Switch to common structure and shared resources
This commit is contained in:
@@ -0,0 +1,80 @@
|
||||
**********************************************
|
||||
Xml format of conditional module [xmodule]
|
||||
**********************************************
|
||||
|
||||
.. module:: conditional_module
|
||||
|
||||
Format description
|
||||
==================
|
||||
|
||||
The main tag of Conditional module input is:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<conditional> ... </conditional>
|
||||
|
||||
``conditional`` can include any number of any xmodule tags (``html``, ``video``, ``poll``, etc.) or ``show`` tags.
|
||||
|
||||
conditional tag
|
||||
---------------
|
||||
|
||||
The main container for a single instance of Conditional module. The following attributes can
|
||||
be specified for this tag::
|
||||
|
||||
sources - location id of required modules, separated by ';'
|
||||
[message | ""] - message for case, where one or more are not passed. Here you can use variable {link}, which generate link to required module.
|
||||
|
||||
[submitted] - map to `is_submitted` module method.
|
||||
(pressing RESET button makes this function to return False.)
|
||||
|
||||
[correct] - map to `is_correct` module method
|
||||
[attempted] - map to `is_attempted` module method
|
||||
[poll_answer] - map to `poll_answer` module attribute
|
||||
[voted] - map to `voted` module attribute
|
||||
|
||||
show tag
|
||||
--------
|
||||
|
||||
Symlink to some set of xmodules. The following attributes can
|
||||
be specified for this tag::
|
||||
|
||||
sources - location id of modules, separated by ';'
|
||||
|
||||
Example
|
||||
=======
|
||||
|
||||
Examples of conditional depends on poll
|
||||
-------------------------------------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<conditional sources="i4x://MITx/0.000x/poll_question/first_real_poll_seq_with_reset" poll_answer="man"
|
||||
message="{link} must be answered for this to become visible.">
|
||||
<html>
|
||||
<h2>You see this, cause your vote value for "First question" was "man"</h2>
|
||||
</html>
|
||||
</conditional>
|
||||
|
||||
Examples of conditional depends on poll (use <show> tag)
|
||||
--------------------------------------------------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<conditional sources="i4x://MITx/0.000x/poll_question/first_real_poll_seq_with_reset" poll_answer="man"
|
||||
message="{link} must be answered for this to become visible.">
|
||||
<html>
|
||||
<show sources="i4x://MITx/0.000x/problem/test_1; i4x://MITx/0.000x/Video/Avi_resources; i4x://MITx/0.000x/problem/test_1"/>
|
||||
</html>
|
||||
</conditional>
|
||||
|
||||
Examples of conditional depends on problem
|
||||
-------------------------------------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<conditional sources="i4x://MITx/0.000x/problem/Conditional:lec27_Q1" attempted="True">
|
||||
<html display_name="HTML for attempted problem">You see this, cause "lec27_Q1" is attempted.</html>
|
||||
</conditional>
|
||||
<conditional sources="i4x://MITx/0.000x/problem/Conditional:lec27_Q1" attempted="False">
|
||||
<html display_name="HTML for not attempted problem">You see this, cause "lec27_Q1" is not attempted.</html>
|
||||
</conditional>
|
||||
779
docs/data/source/course_data_formats/course_xml.rst
Normal file
779
docs/data/source/course_data_formats/course_xml.rst
Normal file
@@ -0,0 +1,779 @@
|
||||
###################
|
||||
Course XML Tutorial
|
||||
###################
|
||||
EdX uses an XML format to describe the structure and contents of its courses. While much of this is abstracted away by the Studio authoring interface, it is still helpful to understand how the edX platform renders a course.
|
||||
|
||||
This guide was written with the assumption that you've dived straight into the edX platform without necessarily having any prior programming/CS knowledge. It will be especially valuable to you if your course is being authored with XML files rather than Studio -- in which case you're likely using functionality that is not yet fully supported in Studio.
|
||||
|
||||
*****
|
||||
Goals
|
||||
*****
|
||||
After reading this, you should be able to:
|
||||
|
||||
* Organize your course content into the files and folders the edX platform expects.
|
||||
* Add new content to a course and make sure it shows up in the courseware.
|
||||
|
||||
*Prerequisites:* it would be helpful to know a little bit about xml. Here is a
|
||||
`simple example <http://www.ultraslavonic.info/intro-to-xml/>`_ if you've never seen it before.
|
||||
|
||||
************
|
||||
Introduction
|
||||
************
|
||||
|
||||
A course is organized hierarchically. We start by describing course-wide parameters, then break the course into chapters, and then go deeper and deeper until we reach a specific pset, video, etc. You could make an analogy to finding a green shirt in your house -> bedroom -> closet -> drawer -> shirts -> green shirt.
|
||||
|
||||
We'll begin with a sample course structure as a case study of how XML and files in a course are organized. More technical details will follow, including discussion of some special cases.
|
||||
|
||||
**********
|
||||
Case Study
|
||||
**********
|
||||
Let's jump right in by looking at the directory structure of a very simple toy course::
|
||||
|
||||
toy/
|
||||
course/
|
||||
course.xml
|
||||
problem/
|
||||
policies/
|
||||
roots/
|
||||
|
||||
The only top level file is `course.xml`, which should contain one line, looking something like this:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<course org="edX" course="toy" url_name="2012_Fall"/>
|
||||
|
||||
This gives all the information to uniquely identify a particular run of any course -- which organization is producing the course, what the course name is, and what "run" this is, specified via the `url_name` attribute.
|
||||
|
||||
Obviously, this doesn't actually specify any of the course content, so we need to find that next. To know where to look, you need to know the standard organizational structure of our system: course elements are uniquely identified by the combination `(category, url_name)`. In this case, we are looking for a `course` element with the `url_name` "2012_Fall". The definition of this element will be in `course/2012_Fall.xml`. Let's look there next::
|
||||
|
||||
toy/
|
||||
course/
|
||||
2012_Fall.xml # <-- Where we look for category="course", url_name="2012_Fall"
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<!-- Contents of course/2012_Fall.xml -->
|
||||
<course>
|
||||
<chapter url_name="Overview">
|
||||
<videosequence url_name="Toy_Videos">
|
||||
<problem url_name="warmup"/>
|
||||
<video url_name="Video_Resources" youtube="1.0:1bK-WdDi6Qw"/>
|
||||
</videosequence>
|
||||
<video url_name="Welcome" youtube="1.0:p2Q6BrNhdh8"/>
|
||||
</chapter>
|
||||
</course>
|
||||
|
||||
Aha. Now we've found some content. We can see that the course is organized hierarchically, in this case with only one chapter, with `url_name` "Overview". The chapter contains a `videosequence` and a `video`, with the sequence containing a problem and another video. When viewed in the courseware, chapters are shown at the top level of the navigation accordion on the left, with any elements directly included in the chapter below.
|
||||
|
||||
Looking at this file, we can see the course structure, and the youtube urls for the videos, but what about the "warmup" problem? There is no problem content here! Where should we look? This is a good time to pause and try to answer that question based on our organizational structure above.
|
||||
|
||||
As you hopefully guessed, the problem would be in `problem/warmup.xml`. (Note: This tutorial doesn't discuss the xml format for problems -- there are chapters of edx4edx that describe it.) This is an instance of a *pointer tag*: any xml tag with only the category and a url_name attribute will point to the file `{category}/{url_name}.xml`. For example, this means that our toy `course.xml` could have also been written as
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<!-- Contents of course/2012_Fall.xml -->
|
||||
<course>
|
||||
<chapter url_name="Overview"/>
|
||||
</course>
|
||||
|
||||
with `chapter/Overview.xml` containing
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<chapter>
|
||||
<videosequence url_name="Toy_Videos">
|
||||
<problem url_name="warmup"/>
|
||||
<video url_name="Video_Resources" youtube="1.0:1bK-WdDi6Qw"/>
|
||||
</videosequence>
|
||||
<video url_name="Welcome" youtube="1.0:p2Q6BrNhdh8"/>
|
||||
</chapter>
|
||||
|
||||
In fact, this is the recommended structure for real courses -- putting each chapter into its own file makes it easy to have different people work on each without conflicting or having to merge. Similarly, as sequences get large, it can be handy to split them out as well (in `sequence/{url_name}.xml`, of course).
|
||||
|
||||
Note that the `url_name` is only specified once per element -- either the inline definition, or in the pointer tag.
|
||||
|
||||
Policy Files
|
||||
============
|
||||
|
||||
We still haven't looked at two of the directories in the top-level listing above: `policies` and `roots`. Let's look at policies next. The policy directory contains one file::
|
||||
|
||||
toy/
|
||||
policies/
|
||||
2012_Fall.json
|
||||
|
||||
and that file is named `{course-url_name}.json`. As you might expect, this file contains a policy for the course. In our example, it looks like this:
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
{
|
||||
"course/2012_Fall": {
|
||||
"graceperiod": "2 days 5 hours 59 minutes 59 seconds",
|
||||
"start": "2015-07-17T12:00",
|
||||
"display_name": "Toy Course"
|
||||
},
|
||||
"chapter/Overview": {
|
||||
"display_name": "Overview"
|
||||
},
|
||||
"videosequence/Toy_Videos": {
|
||||
"display_name": "Toy Videos",
|
||||
"format": "Lecture Sequence"
|
||||
},
|
||||
"problem/warmup": {
|
||||
"display_name": "Getting ready for the semester"
|
||||
},
|
||||
"video/Video_Resources": {
|
||||
"display_name": "Video Resources"
|
||||
},
|
||||
"video/Welcome": {
|
||||
"display_name": "Welcome"
|
||||
}
|
||||
}
|
||||
|
||||
The policy specifies metadata about the content elements -- things which are not inherent to the definition of the content, but which describe how the content is presented to the user and used in the course. See below for a full list of metadata attributes; as the example shows, they include `display_name`, which is what is shown when this piece of content is referenced or shown in the courseware, and various dates and times, like `start`, which specifies when the content becomes visible to students, and various problem-specific parameters like the allowed number of attempts. One important point is that some metadata is inherited: for example, specifying the start date on the course makes it the default for every element in the course. See below for more details.
|
||||
|
||||
It is possible to put metadata directly in the XML, as attributes of the appropriate tag, but using a policy file has two benefits: it puts all the policy in one place, making it easier to check that things like due dates are set properly, and it allows the content definitions to be easily used in another run of the same course, with the same or similar content, but different policy.
|
||||
|
||||
Roots
|
||||
=====
|
||||
The last directory in the top level listing is `roots`. In our toy course, it contains a single file::
|
||||
|
||||
roots/
|
||||
2012_Fall.xml
|
||||
|
||||
This file is identical to the top-level `course.xml`, containing
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<course org="edX" course="toy" url_name="2012_Fall"/>
|
||||
|
||||
In fact, the top level `course.xml` is a symbolic link to this file. When there is only one run of a course, the roots directory is not really necessary, and the top-level course.xml file can just specify the `url_name` of the course. However, if we wanted to make a second run of our toy course, we could add another file called, e.g., `roots/2013_Spring.xml`, containing
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<course org="edX" course="toy" url_name="2013_Spring"/>
|
||||
|
||||
After creating `course/2013_Spring.xml` with the course structure (possibly as a symbolic link or copy of `course/2012_Fall.xml` if no content was changing), and `policies/2013_Spring.json`, we would have two different runs of the toy course in the course repository. Our build system understands this roots structure, and will build a course package for each root.
|
||||
|
||||
.. note::
|
||||
If you're using a local development environment, make the top level `course.xml` point to the desired root, and check out the repo multiple times if you need multiple runs simultaneously).
|
||||
|
||||
That's basically all there is to the organizational structure. Read the next section for details on the tags we support, including some special case tags like `customtag` and `html` invariants, and look at the end for some tips that will make the editing process easier.
|
||||
|
||||
****
|
||||
Tags
|
||||
****
|
||||
|
||||
.. list-table::
|
||||
:widths: 10 80
|
||||
:header-rows: 0
|
||||
|
||||
* - `abtest`
|
||||
- Support for A/B testing. TODO: add details..
|
||||
* - `chapter`
|
||||
- Top level organization unit of a course. The courseware display code currently expects the top level `course` element to contain only chapters, though there is no philosophical reason why this is required, so we may change it to properly display non-chapters at the top level.
|
||||
* - `conditional`
|
||||
- Conditional element, which shows one or more modules only if certain conditions are satisfied.
|
||||
* - `course`
|
||||
- Top level tag. Contains everything else.
|
||||
* - `customtag`
|
||||
- Render an html template, filling in some parameters, and return the resulting html. See below for details.
|
||||
* - `discussion`
|
||||
- Inline discussion forum.
|
||||
* - `html`
|
||||
- A reference to an html file.
|
||||
* - `error`
|
||||
- Don't put these in by hand :) The internal representation of content that has an error, such as malformed XML or some broken invariant.
|
||||
* - `problem`
|
||||
- See elsewhere in edx4edx for documentation on the format.
|
||||
* - `problemset`
|
||||
- Logically, a series of related problems. Currently displayed vertically. May contain explanatory html, videos, etc.
|
||||
* - `sequential`
|
||||
- A sequence of content, currently displayed with a horizontal list of tabs. If possible, use a more semantically meaningful tag (currently, we only have `videosequence`).
|
||||
* - `vertical`
|
||||
- A sequence of content, displayed vertically. Content will be accessed all at once, on the right part of the page. No navigational bar. May have to use browser scroll bars. Content split with separators. If possible, use a more semantically meaningful tag (currently, we only have `problemset`).
|
||||
* - `video`
|
||||
- A link to a video, currently expected to be hosted on youtube.
|
||||
* - `videosequence`
|
||||
- A sequence of videos. This can contain various non-video content; it just signals to the system that this is logically part of an explanatory sequence of content, as opposed to say an exam sequence.
|
||||
|
||||
Container Tags
|
||||
==============
|
||||
Container tags include `chapter`, `sequential`, `videosequence`, `vertical`, and `problemset`. They are all specified in the same way in the xml, as shown in the tutorial above.
|
||||
|
||||
`course`
|
||||
========
|
||||
`course` is also a container, and is similar, with one extra wrinkle: the top level pointer tag *must* have `org` and `course` attributes specified--the organization name, and course name. Note that `course` is referring to the platonic ideal of this course (e.g. "6.002x"), not to any particular run of this course. The `url_name` should be the particular run of this course.
|
||||
|
||||
`conditional`
|
||||
=============
|
||||
`conditional` is as special kind of container tag as well. Here are two examples:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<conditional condition="require_completed" required="problem/choiceprob">
|
||||
<video url_name="secret_video" />
|
||||
</conditional>
|
||||
|
||||
<conditional condition="require_attempted" required="problem/choiceprob&problem/sumprob">
|
||||
<html url_name="secret_page" />
|
||||
</conditional>
|
||||
|
||||
The condition can be either `require_completed`, in which case the required modules must be completed, or `require_attempted`, in which case the required modules must have been attempted.
|
||||
|
||||
The required modules are specified as a set of `tag`/`url_name`, joined by an ampersand.
|
||||
|
||||
`customtag`
|
||||
===========
|
||||
When we see:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<customtag impl="special" animal="unicorn" hat="blue"/>
|
||||
|
||||
We will:
|
||||
|
||||
#. Look for a file called `custom_tags/special` in your course dir.
|
||||
#. Render it as a mako template, passing parameters {'animal':'unicorn', 'hat':'blue'}, generating html. (Google `mako` for template syntax, or look at existing examples).
|
||||
|
||||
Since `customtag` is already a pointer, there is generally no need to put it into a separate file--just use it in place:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<customtag url_name="my_custom_tag" impl="blah" attr1="..."/>
|
||||
|
||||
`discussion`
|
||||
============
|
||||
The discussion tag embeds an inline discussion module. The XML format is:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<discussion for="Course overview" id="6002x_Fall_2012_Overview" discussion_category="Week 1/Overview" />
|
||||
|
||||
The meaning of each attribute is as follows:
|
||||
|
||||
.. list-table::
|
||||
:widths: 10 80
|
||||
:header-rows: 0
|
||||
|
||||
* - `for`
|
||||
- A string that describes the discussion. Purely for descriptive purposes (to the student).
|
||||
* - `id`
|
||||
- The identifier that the discussion forum service uses to refer to this inline discussion module. Since the `id` must be unique and lives in a common namespace with all other courses, the preferred convention is to use `<course_name>_<course_run>_<descriptor>` as in the above example. The `id` should be "machine-friendly", e.g. use alphanumeric characters, underscores. Do **not** use a period (e.g. `6.002x_Fall_2012_Overview`).
|
||||
* - `discussion_category`
|
||||
- The inline module will be indexed in the main "Discussion" tab of the course. The inline discussions are organized into a directory-like hierarchy. Note that the forward slash indicates depth, as in conventional filesytems. In the above example, this discussion module will show up in the following "directory": `Week 1/Overview/Course overview`
|
||||
|
||||
Note that the `for` tag has been appended to the end of the `discussion_category`. This can often lead into deeply nested subforums, which may not be intended. In the above example, if we were to use instead:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<discussion for="Course overview" id="6002x_Fall_2012_Overview" discussion_category="Week 1" />
|
||||
|
||||
This discussion module would show up in the main forums as `Week 1 / Course overview`, which is more succinct.
|
||||
|
||||
`html`
|
||||
======
|
||||
Most of our content is in XML, but some HTML content may not be proper xml (all tags matched, single top-level tag, etc), since browsers are fairly lenient in what they'll display. So, there are two ways to include HTML content:
|
||||
|
||||
* If your HTML content is in a proper XML format, just put it in `html/{url_name}.xml`.
|
||||
* If your HTML content is not in proper XML format, you can put it in `html/{filename}.html`, and put `<html filename={filename} />` in `html/{filename}.xml`. This allows another level of indirection, and makes sure that we can read the XML file and then just return the actual HTML content without trying to parse it.
|
||||
|
||||
`video`
|
||||
=======
|
||||
Videos have an attribute `youtube`, which specifies a series of speeds + youtube video IDs:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<video youtube="0.75:1yk1A8-FPbw,1.0:vNMrbPvwhU4,1.25:gBW_wqe7rDc,1.50:7AE_TKgaBwA"
|
||||
url_name="S15V14_Response_to_impulse_limit_case"/>
|
||||
|
||||
This video has been encoded at 4 different speeds: `0.75x`, `1x`, `1.25x`, and `1.5x`.
|
||||
|
||||
More on `url_name`
|
||||
==================
|
||||
Every content element (within a course) should have a unique id. This id is formed as `{category}/{url_name}`, or automatically generated from the content if `url_name` is not specified. Categories are the different tag types ('chapter', 'problem', 'html', 'sequential', etc). Url_name is a string containing a-z, A-Z, dot (.), underscore (_), and ':'. This is what appears in urls that point to this object.
|
||||
|
||||
Colon (':') is special--when looking for the content definition in an xml, ':' will be replaced with '/'. This allows organizing content into folders. For example, given the pointer tag
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<problem url_name="conceptual:add_apples_and_oranges"/>
|
||||
|
||||
we would look for the problem definition in `problem/conceptual/add_apples_and_oranges.xml`. (There is a technical reason why we can't just allow '/' in the url_name directly.)
|
||||
|
||||
.. important::
|
||||
A student's state for a particular content element is tied to the element ID, so automatic ID generation is only ok for elements that do not need to store any student state (e.g. verticals or customtags). For problems, sequentials, and videos, and any other element where we keep track of what the student has done and where they are at, you should specify a unique `url_name`. Of course, any content element that is split out into a file will need a `url_name` to specify where to find the definition.
|
||||
|
||||
|
||||
************
|
||||
Policy Files
|
||||
************
|
||||
* A policy file is useful when running different versions of a course e.g. internal, external, fall, spring, etc. as you can change due dates, etc, by creating multiple policy files.
|
||||
* A policy file provides information on the metadata of the course--things that are not inherent to the definitions of the contents, but that may vary from run to run.
|
||||
* Note: We will be expanding our understanding and format for metadata in the not-too-distant future, but for now it is simply a set of key-value pairs.
|
||||
|
||||
Locations
|
||||
=========
|
||||
* The policy for a course run `some_url_name` should live in `policies/some_url_name/policy.json` (NOTE: the old format of putting it in `policies/some_url_name.json` will also work, but we suggest using the subdirectory to have all the per-course policy files in one place)
|
||||
* Grading policy files go in `policies/some_url_name/grading_policy.json` (if there's only one course run, can also put it directly in the course root: `/grading_policy.json`)
|
||||
|
||||
Contents
|
||||
========
|
||||
* The file format is JSON, and is best shown by example, as in the tutorial above.
|
||||
* The expected contents are a dictionary mapping from keys to values (syntax `{ key : value, key2 : value2, etc}`)
|
||||
* Keys are in the form `{category}/{url_name}`, which should uniquely identify a content element. Values are dictionaries of the form `{"metadata-key" : "metadata-value"}`.
|
||||
* The order in which things appear does not matter, though it may be helpful to organize the file in the same order as things appear in the content.
|
||||
* NOTE: JSON is picky about commas. If you have trailing commas before closing braces, it will complain and refuse to parse the file. This can be irritating at first.
|
||||
|
||||
Supported fields at the course level
|
||||
====================================
|
||||
|
||||
.. list-table::
|
||||
:widths: 10 80
|
||||
:header-rows: 0
|
||||
|
||||
* - `start`
|
||||
- specify the start date for the course. Format-by-example: `"2012-09-05T12:00"`.
|
||||
* - `advertised_start`
|
||||
- specify what you want displayed as the start date of the course in the course listing and course about pages. This can be useful if you want to let people in early before the formal start. Format-by-example: `"2012-09-05T12:"00`.
|
||||
* - `disable_policy_graph`
|
||||
- set to true (or "Yes"), if the policy graph should be disabled (ie not shown).
|
||||
* - `enrollment_start`, `enrollment_end`
|
||||
- -- when can students enroll? (if not specified, can enroll anytime). Same format as `start`.
|
||||
* - `end`
|
||||
- specify the end date for the course. Format-by-example: `"2012-11-05T12:00"`.
|
||||
* - `end_of_course_survey_url`
|
||||
- a url for an end of course survey -- shown after course is over, next to certificate download links.
|
||||
* - `tabs`
|
||||
- have custom tabs in the courseware. See below for details on config.
|
||||
* - `discussion_blackouts`
|
||||
- An array of time intervals during which you want to disable a student's ability to create or edit posts in the forum. Moderators, Community TAs, and Admins are unaffected. You might use this during exam periods, but please be aware that the forum is often a very good place to catch mistakes and clarify points to students. The better long term solution would be to have better flagging/moderation mechanisms, but this is the hammer we have today. Format by example: `[["2012-10-29T04:00", "2012-11-03T04:00"], ["2012-12-30T04:00", "2013-01-02T04:00"]]`
|
||||
* - `show_calculator`
|
||||
- (value "Yes" if desired)
|
||||
* - `days_early_for_beta`
|
||||
- number of days (floating point ok) early that students in the beta-testers group get to see course content. Can also be specified for any other course element, and overrides values set at higher levels.
|
||||
* - `cohort_config`
|
||||
-
|
||||
* `cohorted` : boolean. Set to true if this course uses student cohorts. If so, all inline discussions are automatically cohorted, and top-level discussion topics are configurable via the cohorted_discussions list. Default is not cohorted).
|
||||
* `cohorted_discussions`: list of discussions that should be cohorted. Any not specified in this list are not cohorted.
|
||||
* `auto_cohort`: Truthy.
|
||||
* `auto_cohort_groups`: `["group name 1", "group name 2", ...]` If `cohorted` and `auto_cohort` is true, automatically put each student into a random group from the `auto_cohort_groups` list, creating the group if needed.
|
||||
* - `pdf_textbooks`
|
||||
- have pdf-based textbooks on tabs in the courseware. See below for details on config.
|
||||
* - `html_textbooks`
|
||||
- have html-based textbooks on tabs in the courseware. See below for details on config.
|
||||
|
||||
|
||||
Available metadata
|
||||
==================
|
||||
|
||||
Not Inherited
|
||||
--------------
|
||||
`display_name`
|
||||
Name that will appear when this content is displayed in the courseware. Useful for all tag types.
|
||||
|
||||
`format`
|
||||
Subheading under display name -- currently only displayed for chapter sub-sections. Also used by the the grader to know how to process students assessments that the section contains. New formats can be defined as a 'type' in the GRADER variable in course_settings.json. Optional. (TODO: double check this--what's the current behavior?)
|
||||
|
||||
`hide_from_toc`
|
||||
If set to true for a chapter or chapter subsection, will hide that element from the courseware navigation accordion. This is useful if you'd like to link to the content directly instead (e.g. for tutorials)
|
||||
|
||||
`ispublic`
|
||||
Specify whether the course is public. You should be able to use start dates instead (?)
|
||||
|
||||
Inherited
|
||||
---------
|
||||
|
||||
`start`
|
||||
When this content should be shown to students. Note that anyone with staff access to the course will always see everything.
|
||||
|
||||
`showanswer`
|
||||
When to show answer. Values: never, attempted, answered, closed, finished, past_due, always. Default: closed. Optional.
|
||||
- `never`: never show answer
|
||||
- `attempted`: show answer after first attempt
|
||||
- `answered` : this is slightly different from `attempted` -- resetting the problems makes "done" False, but leaves attempts unchanged.
|
||||
- `closed` : show answer after problem is closed, ie due date is past, or maximum attempts exceeded.
|
||||
- `finished` : show answer after problem closed, or is correctly answered.
|
||||
- `past_due` : show answer after problem due date is past.
|
||||
- `always` : always allow answer to be shown.
|
||||
|
||||
`graded`
|
||||
Whether this section will count towards the students grade. "true" or "false". Defaults to "false".
|
||||
|
||||
`rerandomize`
|
||||
Randomize question on each attempt. Optional. Possible values:
|
||||
|
||||
`always` (default)
|
||||
Students see a different version of the problem after each attempt to solve it.
|
||||
`onreset`
|
||||
Randomize question when reset button is pressed by the student.
|
||||
`never`
|
||||
All students see the same version of the problem.
|
||||
`per_student`
|
||||
Individual students see the same version of the problem each time the look at it, but that version is different from what other students see.
|
||||
`due`
|
||||
Due date for assignment. Assignment will be closed after that. Values: valid date. Default: none. Optional.
|
||||
`attempts`
|
||||
Number of allowed attempts. Values: integer. Default: infinite. Optional.
|
||||
`graceperiod`
|
||||
A default length of time that the problem is still accessible after the due date in the format `"2 days 3 hours"` or `"1 day 15 minutes"`. Note, graceperiods are currently the easiest way to handle time zones. Due dates are all expressed in UTC.
|
||||
`xqa_key`
|
||||
For integration with Ike's content QA server. -- should typically be specified at the course level.
|
||||
|
||||
Inheritance example
|
||||
-------------------
|
||||
This is a sketch ("tue" is not a valid start date), that should help illustrate how metadata inheritance works.
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<course start="tue">
|
||||
<chap1> -- start tue
|
||||
<problem> --- start tue
|
||||
</chap1>
|
||||
<chap2 start="wed"> -- start wed
|
||||
<problem2 start="thu"> -- start thu
|
||||
<problem3> -- start wed
|
||||
</chap2>
|
||||
</course>
|
||||
|
||||
|
||||
Specifying metadata in the XML file
|
||||
-----------------------------------
|
||||
Metadata can also live in the xml files, but anything defined in the policy file overrides anything in the XML. This is primarily for backwards compatibility, and you should probably not use both. If you do leave some metadata tags in the xml, you should be consistent (e.g. if `display_name` stays in XML, they should all stay in XML. Note `display_name` should be specified in the problem xml definition itself, ie, `<problem display_name="Title">Problem Text</problem>`, in file `ProblemFoo.xml`).
|
||||
|
||||
.. note::
|
||||
Some xml attributes are not metadata. e.g. in `<video youtube="xyz987293487293847"/>`, the `youtube` attribute specifies what video this is, and is logically part of the content, not the policy, so it should stay in the xml.
|
||||
|
||||
Another example policy file::
|
||||
|
||||
{
|
||||
"course/2012": {
|
||||
"graceperiod": "1 day",
|
||||
"start": "2012-10-15T12:00",
|
||||
"display_name": "Introduction to Computer Science I",
|
||||
"xqa_key": "z1y4vdYcy0izkoPeihtPClDxmbY1ogDK"
|
||||
},
|
||||
"chapter/Week_0": {
|
||||
"display_name": "Week 0"
|
||||
},
|
||||
"sequential/Pre-Course_Survey": {
|
||||
"display_name": "Pre-Course Survey",
|
||||
"format": "Survey"
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
Deprecated Formats
|
||||
------------------
|
||||
If you look at some older xml, you may see some tags or metadata attributes that aren't listed above. They are deprecated, and should not be used in new content. We include them here so that you can understand how old-format content works.
|
||||
|
||||
Obsolete Tags
|
||||
^^^^^^^^^^^^^
|
||||
`section`
|
||||
This used to be necessary within chapters. Now, you can just use any standard tag inside a chapter, so use the container tag that makes the most sense for grouping content--e.g. `problemset`, `videosequence`, and just include content directly if it belongs inside a chapter (e.g. `html`, `video`, `problem`)
|
||||
|
||||
`videodev, book, slides, image, discuss`
|
||||
There used to be special purpose tags that all basically did the same thing, and have been subsumed by `customtag`. The list is `videodev, book, slides, image, discuss`. Use `customtag` in new content. (e.g. instead of `<book page="12"/>`, use `<customtag impl="book" page="12"/>`)
|
||||
|
||||
Obsolete Attributes
|
||||
^^^^^^^^^^^^^^^^^^^
|
||||
`slug`
|
||||
Old term for `url_name`. Use `url_name`
|
||||
|
||||
`name`
|
||||
We didn't originally have a distinction between `url_name` and `display_name` -- this made content element ids fragile, so please use `url_name` as a stable unique identifier for the content, and `display_name` as the particular string you'd like to display for it.
|
||||
|
||||
************
|
||||
Static links
|
||||
************
|
||||
If your content links (e.g. in an html file) to `"static/blah/ponies.jpg"`, we will look for this...
|
||||
|
||||
* If your course dir has a `static/` subdirectory, we will look in `YOUR_COURSE_DIR/static/blah/ponies.jpg`. This is the prefered organization, as it does not expose anything except what's in `static/` to the world.
|
||||
* If your course dir does not have a `static/` subdirectory, we will look in `YOUR_COURSE_DIR/blah/ponies.jpg`. This is the old organization, and requires that the web server allow access to everything in the couse dir. To switch to the new organization, move all your static content into a new `static/` dir (e.g. if you currently have things in `images/`, `css/`, and `special/`, create a dir called `static/`, and move `images/, css/, and special/` there).
|
||||
|
||||
Links that include `/course` will be rewritten to the root of your course in the courseware (e.g. `courses/{org}/{course}/{url_name}/` in the current url structure). This is useful for linking to the course wiki, for example.
|
||||
|
||||
****
|
||||
Tabs
|
||||
****
|
||||
|
||||
If you want to customize the courseware tabs displayed for your course, specify a "tabs" list in the course-level policy, like the following example:
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
"tabs" : [
|
||||
{"type": "courseware"},
|
||||
{
|
||||
"type": "course_info",
|
||||
"name": "Course Info"
|
||||
},
|
||||
{
|
||||
"type": "external_link",
|
||||
"name": "My Discussion",
|
||||
"link": "http://www.mydiscussion.org/blah"
|
||||
},
|
||||
{"type": "progress", "name": "Progress"},
|
||||
{"type": "wiki", "name": "Wonderwiki"},
|
||||
{
|
||||
"type": "static_tab",
|
||||
"url_slug": "news",
|
||||
"name": "Exciting news"
|
||||
},
|
||||
{"type": "textbooks"},
|
||||
{"type": "html_textbooks"},
|
||||
{"type": "pdf_textbooks"}
|
||||
]
|
||||
|
||||
* If you specify any tabs, you must specify all tabs. They will appear in the order given.
|
||||
* The first two tabs must have types `"courseware"` and `"course_info"`, in that order, or the course will not load.
|
||||
* The `courseware` tab never has a name attribute -- it's always rendered as "Courseware" for consistency between courses.
|
||||
* The `textbooks` tab will actually generate one tab per textbook, using the textbook titles as names.
|
||||
* The `html_textbooks` tab will actually generate one tab per html_textbook. The tab name is found in the html textbook definition.
|
||||
* The `pdf_textbooks` tab will actually generate one tab per pdf_textbook. The tab name is found in the pdf textbook definition.
|
||||
* For static tabs, the `url_slug` will be the url that points to the tab. It can not be one of the existing courseware url types (even if those aren't used in your course). The static content will come from `tabs/{course_url_name}/{url_slug}.html`, or `tabs/{url_slug}.html` if that doesn't exist.
|
||||
* An Instructor tab will be automatically added at the end for course staff users.
|
||||
|
||||
.. list-table:: Supported Tabs and Parameters
|
||||
:widths: 10 80
|
||||
:header-rows: 0
|
||||
|
||||
* - `courseware`
|
||||
- No other parameters.
|
||||
* - `course_info`
|
||||
- Parameter `name`.
|
||||
* - `wiki`
|
||||
- Parameter `name`.
|
||||
* - `discussion`
|
||||
- Parameter `name`.
|
||||
* - `external_link`
|
||||
- Parameters `name`, `link`.
|
||||
* - `textbooks`
|
||||
- No parameters--generates tab names from book titles.
|
||||
* - `html_textbooks`
|
||||
- No parameters--generates tab names from html book definition. (See discussion below for configuration.)
|
||||
* - `pdf_textbooks`
|
||||
- No parameters--generates tab names from pdf book definition. (See discussion below for configuration.)
|
||||
* - `progress`
|
||||
- Parameter `name`.
|
||||
* - `static_tab`
|
||||
- Parameters `name`, `url_slug`--will look for tab contents in 'tabs/{course_url_name}/{tab url_slug}.html'
|
||||
* - `staff_grading`
|
||||
- No parameters. If specified, displays the staff grading tab for instructors.
|
||||
|
||||
*********
|
||||
Textbooks
|
||||
*********
|
||||
Support is currently provided for image-based, HTML-based and PDF-based textbooks. In addition to enabling the display of textbooks in tabs (see above), specific information about the location of textbook content must be configured.
|
||||
|
||||
Image-based Textbooks
|
||||
=====================
|
||||
|
||||
Configuration
|
||||
-------------
|
||||
|
||||
Image-based textbooks are configured at the course level in the XML markup. Here is an example:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<course>
|
||||
<textbook title="Textbook 1" book_url="https://www.example.com/textbook_1/" />
|
||||
<textbook title="Textbook 2" book_url="https://www.example.com/textbook_2/" />
|
||||
<chapter url_name="Overview">
|
||||
<chapter url_name="First week">
|
||||
</course>
|
||||
|
||||
|
||||
Each `textbook` element is displayed on a different tab. The `title` attribute is used as the tab's name, and the `book_url` attribute points to the remote directory that contains the images of the text. Note the trailing slash on the end of the `book_url` attribute.
|
||||
|
||||
The images must be stored in the same directory as the `book_url`, with filenames matching `pXXX.png`, where `XXX` is a three-digit number representing the page number (with leading zeroes as necessary). Pages start at `p001.png`.
|
||||
|
||||
Each textbook must also have its own table of contents. This is read from the `book_url` location, by appending `toc.xml`. This file contains a `table_of_contents` parent element, with `entry` elements nested below it. Each `entry` has attributes for `name`, `page_label`, and `page`, as well as an optional `chapter` attribute. An arbitrary number of levels of nesting of `entry` elements within other `entry` elements is supported, but you're likely to only want two levels. The `page` represents the actual page to link to, while the `page_label` matches the displayed page number on that page. Here's an example:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<table_of_contents>
|
||||
<entry page="1" page_label="i" name="Title" />
|
||||
<entry page="2" page_label="ii" name="Preamble">
|
||||
<entry page="2" page_label="ii" name="Copyright"/>
|
||||
<entry page="3" page_label="iii" name="Brief Contents"/>
|
||||
<entry page="5" page_label="v" name="Contents"/>
|
||||
<entry page="9" page_label="1" name="About the Authors"/>
|
||||
<entry page="10" page_label="2" name="Acknowledgments"/>
|
||||
<entry page="11" page_label="3" name="Dedication"/>
|
||||
<entry page="12" page_label="4" name="Preface"/>
|
||||
</entry>
|
||||
<entry page="15" page_label="7" name="Introduction to edX" chapter="1">
|
||||
<entry page="15" page_label="7" name="edX in the Modern World"/>
|
||||
<entry page="18" page_label="10" name="The edX Method"/>
|
||||
<entry page="18" page_label="10" name="A Description of edX"/>
|
||||
<entry page="29" page_label="21" name="A Brief History of edX"/>
|
||||
<entry page="51" page_label="43" name="Introduction to edX"/>
|
||||
<entry page="56" page_label="48" name="Endnotes"/>
|
||||
</entry>
|
||||
<entry page="73" page_label="65" name="Art and Photo Credits" chapter="30">
|
||||
<entry page="73" page_label="65" name="Molecular Models"/>
|
||||
<entry page="73" page_label="65" name="Photo Credits"/>
|
||||
</entry>
|
||||
<entry page="77" page_label="69" name="Index" />
|
||||
</table_of_contents>
|
||||
|
||||
|
||||
Linking from Content
|
||||
--------------------
|
||||
|
||||
It is possible to add links to specific pages in a textbook by using a URL that encodes the index of the textbook and the page number. The URL is of the form `/course/book/${bookindex}/$page}`. If the page is omitted from the URL, the first page is assumed.
|
||||
|
||||
You can use a `customtag` to create a template for such links. For example, you can create a `book` template in the `customtag` directory, containing:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<img src="/static/images/icons/textbook_icon.png"/> More information given in <a href="/course/book/${book}/${page}">the text</a>.
|
||||
|
||||
The course content can then link to page 25 using the `customtag` element:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<customtag book="0" page="25" impl="book"/>
|
||||
|
||||
|
||||
HTML-based Textbooks
|
||||
====================
|
||||
|
||||
Configuration
|
||||
-------------
|
||||
|
||||
HTML-based textbooks are configured at the course level in the policy file. The JSON markup consists of an array of maps, with each map corresponding to a separate textbook. There are two styles to presenting HTML-based material. The first way is as a single HTML on a tab, which requires only a tab title and a URL for configuration. A second way permits the display of multiple HTML files that should be displayed together on a single view. For this view, a side panel of links is available on the left, allowing selection of a particular HTML to view.
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
"html_textbooks": [
|
||||
{"tab_title": "Textbook 1",
|
||||
"url": "https://www.example.com/thiscourse/book1/book1.html" },
|
||||
{"tab_title": "Textbook 2",
|
||||
"chapters": [
|
||||
{ "title": "Chapter 1", "url": "https://www.example.com/thiscourse/book2/Chapter1.html" },
|
||||
{ "title": "Chapter 2", "url": "https://www.example.com/thiscourse/book2/Chapter2.html" },
|
||||
{ "title": "Chapter 3", "url": "https://www.example.com/thiscourse/book2/Chapter3.html" },
|
||||
{ "title": "Chapter 4", "url": "https://www.example.com/thiscourse/book2/Chapter4.html" },
|
||||
{ "title": "Chapter 5", "url": "https://www.example.com/thiscourse/book2/Chapter5.html" },
|
||||
{ "title": "Chapter 6", "url": "https://www.example.com/thiscourse/book2/Chapter6.html" },
|
||||
{ "title": "Chapter 7", "url": "https://www.example.com/thiscourse/book2/Chapter7.html" }
|
||||
]
|
||||
}
|
||||
]
|
||||
|
||||
Some notes:
|
||||
|
||||
* It is not a good idea to include a top-level URL and chapter-level URLs in the same textbook configuration.
|
||||
|
||||
Linking from Content
|
||||
--------------------
|
||||
|
||||
It is possible to add links to specific pages in a textbook by using a URL that encodes the index of the textbook, the chapter (if chapters are used), and the page number. For a book with no chapters, the URL is of the form `/course/htmlbook/${bookindex}`. For a book with chapters, use `/course/htmlbook/${bookindex}/chapter/${chapter}` for a specific chapter, or `/course/htmlbook/${bookindex}` will default to the first chapter.
|
||||
|
||||
For example, for the book with no chapters configured above, the textbook can be reached using the URL `/course/htmlbook/0`. Reaching the third chapter of the second book is accomplished with `/course/htmlbook/1/chapter/3`.
|
||||
|
||||
You can use a `customtag` to create a template for such links. For example, you can create a `htmlbook` template in the `customtag` directory, containing:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<img src="/static/images/icons/textbook_icon.png"/> More information given in <a href="/course/htmlbook/${book}">the text</a>.
|
||||
|
||||
And a `htmlchapter` template containing:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<img src="/static/images/icons/textbook_icon.png"/> More information given in <a href="/course/htmlbook/${book}/chapter/${chapter}">the text</a>.
|
||||
|
||||
The example pages can then be linked using the `customtag` element:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<customtag book="0" impl="htmlbook"/>
|
||||
<customtag book="1" chapter="3" impl="htmlchapter"/>
|
||||
|
||||
PDF-based Textbooks
|
||||
===================
|
||||
|
||||
Configuration
|
||||
-------------
|
||||
|
||||
PDF-based textbooks are configured at the course level in the policy file. The JSON markup consists of an array of maps, with each map corresponding to a separate textbook. There are two styles to presenting PDF-based material. The first way is as a single PDF on a tab, which requires only a tab title and a URL for configuration. A second way permits the display of multiple PDFs that should be displayed together on a single view. For this view, a side panel of links is available on the left, allowing selection of a particular PDF to view.
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
"pdf_textbooks": [
|
||||
{"tab_title": "Textbook 1",
|
||||
"url": "https://www.example.com/thiscourse/book1/book1.pdf" },
|
||||
{"tab_title": "Textbook 2",
|
||||
"chapters": [
|
||||
{ "title": "Chapter 1", "url": "https://www.example.com/thiscourse/book2/Chapter1.pdf" },
|
||||
{ "title": "Chapter 2", "url": "https://www.example.com/thiscourse/book2/Chapter2.pdf" },
|
||||
{ "title": "Chapter 3", "url": "https://www.example.com/thiscourse/book2/Chapter3.pdf" },
|
||||
{ "title": "Chapter 4", "url": "https://www.example.com/thiscourse/book2/Chapter4.pdf" },
|
||||
{ "title": "Chapter 5", "url": "https://www.example.com/thiscourse/book2/Chapter5.pdf" },
|
||||
{ "title": "Chapter 6", "url": "https://www.example.com/thiscourse/book2/Chapter6.pdf" },
|
||||
{ "title": "Chapter 7", "url": "https://www.example.com/thiscourse/book2/Chapter7.pdf" }
|
||||
]
|
||||
}
|
||||
]
|
||||
|
||||
Some notes:
|
||||
|
||||
* It is not a good idea to include a top-level URL and chapter-level URLs in the same textbook configuration.
|
||||
|
||||
Linking from Content
|
||||
--------------------
|
||||
|
||||
It is possible to add links to specific pages in a textbook by using a URL that encodes the index of the textbook, the chapter (if chapters are used), and the page number. For a book with no chapters, the URL is of the form `/course/pdfbook/${bookindex}/$page}`. For a book with chapters, use `/course/pdfbook/${bookindex}/chapter/${chapter}/${page}`. If the page is omitted from the URL, the first page is assumed.
|
||||
|
||||
For example, for the book with no chapters configured above, page 25 can be reached using the URL `/course/pdfbook/0/25`. Reaching page 19 in the third chapter of the second book is accomplished with `/course/pdfbook/1/chapter/3/19`.
|
||||
|
||||
You can use a `customtag` to create a template for such links. For example, you can create a `pdfbook` template in the `customtag` directory, containing:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<img src="/static/images/icons/textbook_icon.png"/> More information given in <a href="/course/pdfbook/${book}/${page}">the text</a>.
|
||||
|
||||
And a `pdfchapter` template containing:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<img src="/static/images/icons/textbook_icon.png"/> More information given in <a href="/course/pdfbook/${book}/chapter/${chapter}/${page}">the text</a>.
|
||||
|
||||
The example pages can then be linked using the `customtag` element:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<customtag book="0" page="25" impl="pdfbook"/>
|
||||
<customtag book="1" chapter="3" page="19" impl="pdfchapter"/>
|
||||
|
||||
|
||||
*************************************
|
||||
Other file locations (info and about)
|
||||
*************************************
|
||||
With different course runs, we may want different course info and about materials. This is now supported by putting files in as follows::
|
||||
|
||||
/
|
||||
about/
|
||||
foo.html -- shared default for all runs
|
||||
url_name1/
|
||||
foo.html -- version used for url_name1
|
||||
bar.html -- bar for url_name1
|
||||
url_name2/
|
||||
bar.html -- bar for url_name2
|
||||
-- url_name2 will use default foo.html
|
||||
|
||||
and the same works for the `info` directory.
|
||||
|
||||
|
||||
***************************
|
||||
Tips for content developers
|
||||
***************************
|
||||
|
||||
#. We will be making better tools for managing policy files soon. In the meantime, you can add dummy definitions to make it easier to search and separate the file visually. For example, you could add `"WEEK 1" : "###################"`, before the week 1 material to make it easy to find in the file.
|
||||
|
||||
#. Come up with a consistent pattern for url_names, so that it's easy to know where to look for any piece of content. It will also help to come up with a standard way of splitting your content files. As a point of departure, we suggest splitting chapters, sequences, html, and problems into separate files.
|
||||
|
||||
#. Prefer the most "semantic" name for containers: e.g., use problemset rather than sequential for a problem set. That way, if we decide to display problem sets differently, we don't have to change the XML.
|
||||
|
||||
142
docs/data/source/course_data_formats/custom_response.rst
Normal file
142
docs/data/source/course_data_formats/custom_response.rst
Normal file
@@ -0,0 +1,142 @@
|
||||
####################################
|
||||
CustomResponse XML and Python Script
|
||||
####################################
|
||||
|
||||
This document explains how to write a CustomResponse problem. CustomResponse
|
||||
problems execute Python script to check student answers and provide hints.
|
||||
|
||||
There are two general ways to create a CustomResponse problem:
|
||||
|
||||
|
||||
*****************
|
||||
Answer tag format
|
||||
*****************
|
||||
One format puts the Python code in an ``<answer>`` tag:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<problem>
|
||||
<p>What is the sum of 2 and 3?</p>
|
||||
|
||||
<customresponse expect="5">
|
||||
<textline math="1" />
|
||||
</customresponse>
|
||||
|
||||
<answer>
|
||||
# Python script goes here
|
||||
</answer>
|
||||
</problem>
|
||||
|
||||
|
||||
The Python script interacts with these variables in the global context:
|
||||
* ``answers``: An ordered list of answers the student provided.
|
||||
For example, if the student answered ``6``, then ``answers[0]`` would
|
||||
equal ``6``.
|
||||
* ``expect``: The value of the ``expect`` attribute of ``<customresponse>``
|
||||
(if provided).
|
||||
* ``correct``: An ordered list of strings indicating whether the
|
||||
student answered the question correctly. Valid values are
|
||||
``"correct"``, ``"incorrect"``, and ``"unknown"``. You can set these
|
||||
values in the script.
|
||||
* ``messages``: An ordered list of message strings that will be displayed
|
||||
beneath each input. You can use this to provide hints to users.
|
||||
For example ``messages[0] = "The capital of California is Sacramento"``
|
||||
would display that message beneath the first input of the response.
|
||||
* ``overall_message``: A string that will be displayed beneath the
|
||||
entire problem. You can use this to provide a hint that applies
|
||||
to the entire problem rather than a particular input.
|
||||
|
||||
Example of a checking script:
|
||||
|
||||
.. code-block:: python
|
||||
|
||||
if answers[0] == expect:
|
||||
correct[0] = 'correct'
|
||||
overall_message = 'Good job!'
|
||||
else:
|
||||
correct[0] = 'incorrect'
|
||||
messages[0] = 'This answer is incorrect'
|
||||
overall_message = 'Please try again'
|
||||
|
||||
**Important**: Python is picky about indentation. Within the ``<answer>`` tag,
|
||||
you must begin your script with no indentation.
|
||||
|
||||
*****************
|
||||
Script tag format
|
||||
*****************
|
||||
The other way to create a CustomResponse is to put a "checking function"
|
||||
in a ``<script>`` tag, then use the ``cfn`` attribute of the
|
||||
``<customresponse>`` tag:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<problem>
|
||||
<p>What is the sum of 2 and 3?</p>
|
||||
|
||||
<customresponse cfn="check_func" expect="5">
|
||||
<textline math="1" />
|
||||
</customresponse>
|
||||
|
||||
<script type="loncapa/python">
|
||||
def check_func(expect, ans):
|
||||
# Python script goes here
|
||||
</script>
|
||||
</problem>
|
||||
|
||||
|
||||
**Important**: Python is picky about indentation. Within the ``<script>`` tag,
|
||||
the ``def check_func(expect, ans):`` line must have no indentation.
|
||||
|
||||
The check function accepts two arguments:
|
||||
* ``expect`` is the value of the ``expect`` attribute of ``<customresponse>``
|
||||
(if provided)
|
||||
* ``answer`` is either:
|
||||
|
||||
* The value of the answer the student provided, if there is only one input.
|
||||
* An ordered list of answers the student provided, if there
|
||||
are multiple inputs.
|
||||
|
||||
There are several ways that the check function can indicate whether the student
|
||||
succeeded. The check function can return any of the following:
|
||||
|
||||
* ``True``: Indicates that the student answered correctly for all inputs.
|
||||
* ``False``: Indicates that the student answered incorrectly.
|
||||
All inputs will be marked incorrect.
|
||||
* A dictionary of the form: ``{ 'ok': True, 'msg': 'Message' }``
|
||||
If the dictionary's value for ``ok`` is set to ``True``, all inputs are
|
||||
marked correct; if it is set to ``False``, all inputs are marked incorrect.
|
||||
The ``msg`` is displayed beneath all inputs, and it may contain
|
||||
XHTML markup.
|
||||
* A dictionary of the form
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
|
||||
{ 'overall_message': 'Overall message',
|
||||
'input_list': [
|
||||
{ 'ok': True, 'msg': 'Feedback for input 1'},
|
||||
{ 'ok': False, 'msg': 'Feedback for input 2'},
|
||||
... ] }
|
||||
|
||||
The last form is useful for responses that contain multiple inputs.
|
||||
It allows you to provide feedback for each input individually,
|
||||
as well as a message that applies to the entire response.
|
||||
|
||||
Example of a checking function:
|
||||
|
||||
.. code-block:: python
|
||||
|
||||
def check_func(expect, answer_given):
|
||||
check1 = (int(answer_given[0]) == 1)
|
||||
check2 = (int(answer_given[1]) == 2)
|
||||
check3 = (int(answer_given[2]) == 3)
|
||||
return {'overall_message': 'Overall message',
|
||||
'input_list': [
|
||||
{ 'ok': check1, 'msg': 'Feedback 1'},
|
||||
{ 'ok': check2, 'msg': 'Feedback 2'},
|
||||
{ 'ok': check3, 'msg': 'Feedback 3'} ] }
|
||||
|
||||
The function checks that the user entered ``1`` for the first input,
|
||||
``2`` for the second input, and ``3`` for the third input.
|
||||
It provides feedback messages for each individual input, as well
|
||||
as a message displayed beneath the entire problem.
|
||||
@@ -0,0 +1,526 @@
|
||||
<problem display_name="Drag and drop demos: drag and drop icons or labels
|
||||
to proper positions." >
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Anyof rule example]</h4><br/>
|
||||
<h4>Please label hydrogen atoms connected with left carbon atom.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/ethglycol.jpg" target_outline="true"
|
||||
one_per_target="true" no_labels="true" label_bg_color="rgb(222, 139, 238)">
|
||||
<draggable id="1" label="Hydrogen" />
|
||||
<draggable id="2" label="Hydrogen" />
|
||||
|
||||
<target id="t1_o" x="10" y="67" w="100" h="100"/>
|
||||
<target id="t2" x="133" y="3" w="70" h="70"/>
|
||||
<target id="t3" x="2" y="384" w="70" h="70"/>
|
||||
<target id="t4" x="95" y="386" w="70" h="70"/>
|
||||
<target id="t5_c" x="94" y="293" w="91" h="91"/>
|
||||
<target id="t6_c" x="328" y="294" w="91" h="91"/>
|
||||
<target id="t7" x="393" y="463" w="70" h="70"/>
|
||||
<target id="t8" x="344" y="214" w="70" h="70"/>
|
||||
<target id="t9_o" x="445" y="162" w="100" h="100"/>
|
||||
<target id="t10" x="591" y="132" w="70" h="70"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{'draggables': ['1', '2'],
|
||||
'targets': ['t2', 't3', 't4' ],
|
||||
'rule':'anyof'
|
||||
}]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Complex grading example]</h4><br/>
|
||||
<h4>Describe carbon molecule in LCAO-MO.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/lcao-mo/lcao-mo.jpg" target_outline="true" >
|
||||
|
||||
<!-- filled bond -->
|
||||
<draggable id="1" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="2" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="3" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="4" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="5" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="6" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
|
||||
<!-- up bond -->
|
||||
<draggable id="7" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="8" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="9" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="10" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
|
||||
<!-- sigma -->
|
||||
<draggable id="11" icon="/static/images/images_list/lcao-mo/sigma.png"/>
|
||||
<draggable id="12" icon="/static/images/images_list/lcao-mo/sigma.png"/>
|
||||
|
||||
<!-- sigma* -->
|
||||
<draggable id="13" icon="/static/images/images_list/lcao-mo/sigma_s.png"/>
|
||||
<draggable id="14" icon="/static/images/images_list/lcao-mo/sigma_s.png"/>
|
||||
|
||||
<!-- pi -->
|
||||
<draggable id="15" icon="/static/images/images_list/lcao-mo/pi.png" />
|
||||
|
||||
<!-- pi* -->
|
||||
<draggable id="16" icon="/static/images/images_list/lcao-mo/pi_s.png" />
|
||||
|
||||
<!-- images that should not be dragged -->
|
||||
<draggable id="17" icon="/static/images/images_list/lcao-mo/d.png" />
|
||||
<draggable id="18" icon="/static/images/images_list/lcao-mo/d.png" />
|
||||
|
||||
<!-- positions of electrons and electron pairs -->
|
||||
<target id="s_left" x="130" y="360" w="32" h="32"/>
|
||||
<target id="s_right" x="505" y="360" w="32" h="32"/>
|
||||
<target id="s_sigma" x="320" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star" x="320" y="290" w="32" h="32"/>
|
||||
<target id="p_left_1" x="80" y="100" w="32" h="32"/>
|
||||
<target id="p_left_2" x="125" y="100" w="32" h="32"/>
|
||||
<target id="p_left_3" x="175" y="100" w="32" h="32"/>
|
||||
<target id="p_right_1" x="465" y="100" w="32" h="32"/>
|
||||
<target id="p_right_2" x="515" y="100" w="32" h="32"/>
|
||||
<target id="p_right_3" x="560" y="100" w="32" h="32"/>
|
||||
<target id="p_pi_1" x="290" y="220" w="32" h="32"/>
|
||||
<target id="p_pi_2" x="335" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma" x="315" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_1" x="290" y="40" w="32" h="32"/>
|
||||
<target id="p_pi_star_2" x="340" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star" x="315" y="0" w="32" h="32"/>
|
||||
|
||||
<!-- positions of names of energy levels -->
|
||||
<target id="s_sigma_name" x="400" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star_name" x="400" y="290" w="32" h="32"/>
|
||||
<target id="p_pi_name" x="400" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma_name" x="400" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_name" x="400" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star_name" x="400" y="0" w="32" h="32"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['1', '2', '3', '4', '5', '6'],
|
||||
'targets': [
|
||||
's_left', 's_right', 's_sigma', 's_sigma_star', 'p_pi_1', 'p_pi_2'
|
||||
],
|
||||
'rule': 'unordered_equal'
|
||||
}, {
|
||||
'draggables': ['7','8', '9', '10'],
|
||||
'targets': ['p_left_1', 'p_left_2', 'p_right_1','p_right_2'],
|
||||
'rule': 'unordered_equal'
|
||||
}, {
|
||||
'draggables': ['11', '12'],
|
||||
'targets': ['s_sigma_name', 'p_sigma_name'],
|
||||
'rule': 'unordered_equal'
|
||||
}, {
|
||||
'draggables': ['13', '14'],
|
||||
'targets': ['s_sigma_star_name', 'p_sigma_star_name'],
|
||||
'rule': 'unordered_equal'
|
||||
}, {
|
||||
'draggables': ['15'],
|
||||
'targets': ['p_pi_name'],
|
||||
'rule': 'unordered_equal'
|
||||
}, {
|
||||
'draggables': ['16'],
|
||||
'targets': ['p_pi_star_name'],
|
||||
'rule': 'unordered_equal'
|
||||
}]
|
||||
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Another complex grading example]</h4><br/>
|
||||
<h4>Describe oxygen molecule in LCAO-MO</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/lcao-mo/lcao-mo.jpg" target_outline="true" one_per_target="true">
|
||||
<!-- filled bond -->
|
||||
<draggable id="1" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="2" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="3" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="4" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="5" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="6" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="v_fb_1" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="v_fb_2" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
<draggable id="v_fb_3" icon="/static/images/images_list/lcao-mo/u_d.png" />
|
||||
|
||||
<!-- up bond -->
|
||||
<draggable id="7" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="8" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="9" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="10" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="v_ub_1" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
<draggable id="v_ub_2" icon="/static/images/images_list/lcao-mo/up.png"/>
|
||||
|
||||
<!-- sigma -->
|
||||
<draggable id="11" icon="/static/images/images_list/lcao-mo/sigma.png"/>
|
||||
<draggable id="12" icon="/static/images/images_list/lcao-mo/sigma.png"/>
|
||||
|
||||
<!-- sigma* -->
|
||||
<draggable id="13" icon="/static/images/images_list/lcao-mo/sigma_s.png"/>
|
||||
<draggable id="14" icon="/static/images/images_list/lcao-mo/sigma_s.png"/>
|
||||
|
||||
<!-- pi -->
|
||||
<draggable id="15" icon="/static/images/images_list/lcao-mo/pi.png" />
|
||||
|
||||
<!-- pi* -->
|
||||
<draggable id="16" icon="/static/images/images_list/lcao-mo/pi_s.png" />
|
||||
|
||||
<!-- images that should not be dragged -->
|
||||
<draggable id="17" icon="/static/images/images_list/lcao-mo/d.png" />
|
||||
<draggable id="18" icon="/static/images/images_list/lcao-mo/d.png" />
|
||||
|
||||
<!-- positions of electrons and electron pairs -->
|
||||
<target id="s_left" x="130" y="360" w="32" h="32"/>
|
||||
<target id="s_right" x="505" y="360" w="32" h="32"/>
|
||||
<target id="s_sigma" x="320" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star" x="320" y="290" w="32" h="32"/>
|
||||
<target id="p_left_1" x="80" y="100" w="32" h="32"/>
|
||||
<target id="p_left_2" x="125" y="100" w="32" h="32"/>
|
||||
<target id="p_left_3" x="175" y="100" w="32" h="32"/>
|
||||
<target id="p_right_1" x="465" y="100" w="32" h="32"/>
|
||||
<target id="p_right_2" x="515" y="100" w="32" h="32"/>
|
||||
<target id="p_right_3" x="560" y="100" w="32" h="32"/>
|
||||
<target id="p_pi_1" x="290" y="220" w="32" h="32"/>
|
||||
<target id="p_pi_2" x="335" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma" x="315" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_1" x="290" y="40" w="32" h="32"/>
|
||||
<target id="p_pi_star_2" x="340" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star" x="315" y="0" w="32" h="32"/>
|
||||
|
||||
<!-- positions of names of energy levels -->
|
||||
<target id="s_sigma_name" x="400" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star_name" x="400" y="290" w="32" h="32"/>
|
||||
<target id="p_pi_name" x="400" y="220" w="32" h="32"/>
|
||||
<target id="p_pi_star_name" x="400" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_name" x="400" y="170" w="32" h="32"/>
|
||||
<target id="p_sigma_star_name" x="400" y="0" w="32" h="32"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [{
|
||||
'draggables': ['1', '2', '3', '4', '5', '6', 'v_fb_1', 'v_fb_2', 'v_fb_3'],
|
||||
'targets': [
|
||||
's_left', 's_right', 's_sigma', 's_sigma_star', 'p_pi_1', 'p_pi_2',
|
||||
'p_sigma', 'p_left_1', 'p_right_3'
|
||||
],
|
||||
'rule': 'anyof'
|
||||
}, {
|
||||
'draggables': ['7', '8', '9', '10', 'v_ub_1', 'v_ub_2'],
|
||||
'targets': [
|
||||
'p_left_2', 'p_left_3', 'p_right_1', 'p_right_2', 'p_pi_star_1',
|
||||
'p_pi_star_2'
|
||||
],
|
||||
'rule': 'anyof'
|
||||
}, {
|
||||
'draggables': ['11', '12'],
|
||||
'targets': ['s_sigma_name', 'p_sigma_name'],
|
||||
'rule': 'anyof'
|
||||
}, {
|
||||
'draggables': ['13', '14'],
|
||||
'targets': ['s_sigma_star_name', 'p_sigma_star_name'],
|
||||
'rule': 'anyof'
|
||||
}, {
|
||||
'draggables': ['15'],
|
||||
'targets': ['p_pi_name'],
|
||||
'rule': 'anyof'
|
||||
}, {
|
||||
'draggables': ['16'],
|
||||
'targets': ['p_pi_star_name'],
|
||||
'rule': 'anyof'
|
||||
}]
|
||||
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Individual targets with outlines, One draggable per target]</h4><br/>
|
||||
<h4>
|
||||
Drag -Ant- to first position and -Star- to third position </h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" target_outline="true">
|
||||
<draggable id="1" label="Label 1"/>
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg"/>
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" />
|
||||
<draggable id="5" label="Label2" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" />
|
||||
<draggable id="7" label="Label3" />
|
||||
|
||||
<target id="t1" x="20" y="20" w="90" h="90"/>
|
||||
<target id="t2" x="300" y="100" w="90" h="90"/>
|
||||
<target id="t3" x="150" y="40" w="50" h="50"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = {'name_with_icon': 't1', 'name4': 't2'}
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[SMALL IMAGE, Individual targets WITHOUT outlines, One draggable
|
||||
per target]</h4><br/>
|
||||
<h4>
|
||||
Move -Star- to the volcano opening, and -Label3- on to
|
||||
the right ear of the cow.
|
||||
</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow3.png" target_outline="false">
|
||||
<draggable id="1" label="Label 1"/>
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg"/>
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" />
|
||||
<draggable id="5" label="Label2" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" />
|
||||
<draggable id="7" label="Label3" />
|
||||
|
||||
<target id="t1" x="111" y="58" w="90" h="90"/>
|
||||
<target id="t2" x="212" y="90" w="90" h="90"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = {'name4': 't1',
|
||||
'7': 't2'}
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Many draggables per target]</h4><br/>
|
||||
<h4>Move -Star- and -Ant- to most left target
|
||||
and -Label3- and -Label2- to most right target.</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" target_outline="true" one_per_target="false">
|
||||
<draggable id="1" label="Label 1"/>
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg"/>
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" />
|
||||
<draggable id="5" label="Label2" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" />
|
||||
<draggable id="7" label="Label3" />
|
||||
|
||||
<target id="t1" x="20" y="20" w="90" h="90"/>
|
||||
<target id="t2" x="300" y="100" w="90" h="90"/>
|
||||
<target id="t3" x="150" y="40" w="50" h="50"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = {'name4': 't1',
|
||||
'name_with_icon': 't1',
|
||||
'5': 't2',
|
||||
'7':'t2'}
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Draggables can be placed anywhere on base image]</h4><br/>
|
||||
<h4>
|
||||
Place -Grass- in the middle of the image and -Ant- in the
|
||||
right upper corner.</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" >
|
||||
<draggable id="1" label="Label 1"/>
|
||||
<draggable id="ant" label="Ant" icon="/static/images/images_list/ant.jpg"/>
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" />
|
||||
<draggable id="5" label="Label2" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" />
|
||||
<draggable id="grass" label="Grass" icon="/static/images/images_list/grass.jpg" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" />
|
||||
<draggable id="7" label="Label3" />
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = {'grass': [[300, 200], 200],
|
||||
'ant': [[500, 0], 200]}
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Another anyof example]</h4><br/>
|
||||
<h4>Please identify the Carbon and Oxygen atoms in the molecule.</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/ethglycol.jpg" target_outline="true" one_per_target="true">
|
||||
<draggable id="l1_c" label="Carbon" />
|
||||
<draggable id="l2" label="Methane"/>
|
||||
<draggable id="l3_o" label="Oxygen" />
|
||||
<draggable id="l4" label="Calcium" />
|
||||
<draggable id="l5" label="Methane"/>
|
||||
<draggable id="l6" label="Calcium" />
|
||||
<draggable id="l7" label="Hydrogen" />
|
||||
<draggable id="l8_c" label="Carbon" />
|
||||
<draggable id="l9" label="Hydrogen" />
|
||||
<draggable id="l10_o" label="Oxygen" />
|
||||
|
||||
<target id="t1_o" x="10" y="67" w="100" h="100"/>
|
||||
<target id="t2" x="133" y="3" w="70" h="70"/>
|
||||
<target id="t3" x="2" y="384" w="70" h="70"/>
|
||||
<target id="t4" x="95" y="386" w="70" h="70"/>
|
||||
<target id="t5_c" x="94" y="293" w="91" h="91"/>
|
||||
<target id="t6_c" x="328" y="294" w="91" h="91"/>
|
||||
<target id="t7" x="393" y="463" w="70" h="70"/>
|
||||
<target id="t8" x="344" y="214" w="70" h="70"/>
|
||||
<target id="t9_o" x="445" y="162" w="100" h="100"/>
|
||||
<target id="t10" x="591" y="132" w="70" h="70"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['l3_o', 'l10_o'],
|
||||
'targets': ['t1_o', 't9_o'],
|
||||
'rule': 'anyof'
|
||||
},
|
||||
{
|
||||
'draggables': ['l1_c','l8_c'],
|
||||
'targets': ['t5_c','t6_c'],
|
||||
'rule': 'anyof'
|
||||
}
|
||||
]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Again another anyof example]</h4><br/>
|
||||
<h4>If the element appears in this molecule, drag the label onto it</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/ethglycol.jpg" target_outline="true"
|
||||
one_per_target="true" no_labels="true" label_bg_color="rgb(222, 139, 238)">
|
||||
<draggable id="1" label="Hydrogen" />
|
||||
<draggable id="2" label="Hydrogen" />
|
||||
<draggable id="3" label="Nytrogen" />
|
||||
<draggable id="4" label="Nytrogen" />
|
||||
<draggable id="5" label="Boron" />
|
||||
<draggable id="6" label="Boron" />
|
||||
<draggable id="7" label="Carbon" />
|
||||
<draggable id="8" label="Carbon" />
|
||||
|
||||
<target id="t1_o" x="10" y="67" w="100" h="100"/>
|
||||
<target id="t2_h" x="133" y="3" w="70" h="70"/>
|
||||
<target id="t3_h" x="2" y="384" w="70" h="70"/>
|
||||
<target id="t4_h" x="95" y="386" w="70" h="70"/>
|
||||
<target id="t5_c" x="94" y="293" w="91" h="91"/>
|
||||
<target id="t6_c" x="328" y="294" w="91" h="91"/>
|
||||
<target id="t7_h" x="393" y="463" w="70" h="70"/>
|
||||
<target id="t8_h" x="344" y="214" w="70" h="70"/>
|
||||
<target id="t9_o" x="445" y="162" w="100" h="100"/>
|
||||
<target id="t10_h" x="591" y="132" w="70" h="70"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['7', '8'],
|
||||
'targets': ['t5_c', 't6_c'],
|
||||
'rule': 'anyof'
|
||||
},
|
||||
{
|
||||
'draggables': ['1', '2'],
|
||||
'targets': ['t2_h', 't3_h', 't4_h', 't7_h', 't8_h', 't10_h'],
|
||||
'rule': 'anyof'
|
||||
}]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Wrong base image url example]
|
||||
</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow3_bad.png" target_outline="false">
|
||||
<draggable id="1" label="Label 1"/>
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg"/>
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" />
|
||||
<draggable id="5" label="Label2" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" />
|
||||
<draggable id="7" label="Label3" />
|
||||
|
||||
<target id="t1" x="111" y="58" w="90" h="90"/>
|
||||
<target id="t2" x="212" y="90" w="90" h="90"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = {'name4': 't1',
|
||||
'7': 't2'}
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
</problem>
|
||||
@@ -0,0 +1,373 @@
|
||||
<problem display_name="Drag and drop demos: drag and drop icons or labels
|
||||
to proper positions." >
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Draggable is reusable example]</h4>
|
||||
<br/>
|
||||
<h4>Please label all hydrogen atoms.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input
|
||||
img="/static/images/images_list/ethglycol.jpg"
|
||||
target_outline="true"
|
||||
one_per_target="true"
|
||||
no_labels="true"
|
||||
label_bg_color="rgb(222, 139, 238)"
|
||||
>
|
||||
<draggable id="1" label="Hydrogen" can_reuse='true' />
|
||||
|
||||
<target id="t1_o" x="10" y="67" w="100" h="100" />
|
||||
<target id="t2" x="133" y="3" w="70" h="70" />
|
||||
<target id="t3" x="2" y="384" w="70" h="70" />
|
||||
<target id="t4" x="95" y="386" w="70" h="70" />
|
||||
<target id="t5_c" x="94" y="293" w="91" h="91" />
|
||||
<target id="t6_c" x="328" y="294" w="91" h="91" />
|
||||
<target id="t7" x="393" y="463" w="70" h="70" />
|
||||
<target id="t8" x="344" y="214" w="70" h="70" />
|
||||
<target id="t9_o" x="445" y="162" w="100" h="100" />
|
||||
<target id="t10" x="591" y="132" w="70" h="70" />
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python">
|
||||
<![CDATA[
|
||||
correct_answer = [{
|
||||
'draggables': ['1'],
|
||||
'targets': ['t2', 't3', 't4', 't7', 't8', 't10'],
|
||||
'rule': 'exact'
|
||||
}]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]>
|
||||
</answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Complex grading example]</h4><br/>
|
||||
<h4>Describe carbon molecule in LCAO-MO.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/lcao-mo/lcao-mo.jpg" target_outline="true" >
|
||||
|
||||
<!-- filled bond -->
|
||||
<draggable id="1" icon="/static/images/images_list/lcao-mo/u_d.png" can_reuse="true" />
|
||||
|
||||
<!-- up bond -->
|
||||
<draggable id="7" icon="/static/images/images_list/lcao-mo/up.png" can_reuse="true" />
|
||||
|
||||
<!-- sigma -->
|
||||
<draggable id="11" icon="/static/images/images_list/lcao-mo/sigma.png" can_reuse="true" />
|
||||
|
||||
<!-- sigma* -->
|
||||
<draggable id="13" icon="/static/images/images_list/lcao-mo/sigma_s.png" can_reuse="true" />
|
||||
|
||||
<!-- pi -->
|
||||
<draggable id="15" icon="/static/images/images_list/lcao-mo/pi.png" can_reuse="true" />
|
||||
|
||||
<!-- pi* -->
|
||||
<draggable id="16" icon="/static/images/images_list/lcao-mo/pi_s.png" can_reuse="true" />
|
||||
|
||||
<!-- images that should not be dragged -->
|
||||
<draggable id="17" icon="/static/images/images_list/lcao-mo/d.png" can_reuse="true" />
|
||||
|
||||
<!-- positions of electrons and electron pairs -->
|
||||
<target id="s_left" x="130" y="360" w="32" h="32"/>
|
||||
<target id="s_right" x="505" y="360" w="32" h="32"/>
|
||||
<target id="s_sigma" x="320" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star" x="320" y="290" w="32" h="32"/>
|
||||
<target id="p_left_1" x="80" y="100" w="32" h="32"/>
|
||||
<target id="p_left_2" x="125" y="100" w="32" h="32"/>
|
||||
<target id="p_left_3" x="175" y="100" w="32" h="32"/>
|
||||
<target id="p_right_1" x="465" y="100" w="32" h="32"/>
|
||||
<target id="p_right_2" x="515" y="100" w="32" h="32"/>
|
||||
<target id="p_right_3" x="560" y="100" w="32" h="32"/>
|
||||
<target id="p_pi_1" x="290" y="220" w="32" h="32"/>
|
||||
<target id="p_pi_2" x="335" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma" x="315" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_1" x="290" y="40" w="32" h="32"/>
|
||||
<target id="p_pi_star_2" x="340" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star" x="315" y="0" w="32" h="32"/>
|
||||
|
||||
<!-- positions of names of energy levels -->
|
||||
<target id="s_sigma_name" x="400" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star_name" x="400" y="290" w="32" h="32"/>
|
||||
<target id="p_pi_name" x="400" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma_name" x="400" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_name" x="400" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star_name" x="400" y="0" w="32" h="32"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['1'],
|
||||
'targets': [
|
||||
's_left', 's_right', 's_sigma', 's_sigma_star', 'p_pi_1', 'p_pi_2'
|
||||
],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['7'],
|
||||
'targets': ['p_left_1', 'p_left_2', 'p_right_1','p_right_2'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['11'],
|
||||
'targets': ['s_sigma_name', 'p_sigma_name'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['13'],
|
||||
'targets': ['s_sigma_star_name', 'p_sigma_star_name'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['15'],
|
||||
'targets': ['p_pi_name'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['16'],
|
||||
'targets': ['p_pi_star_name'],
|
||||
'rule': 'exact'
|
||||
}]
|
||||
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Many draggables per target]</h4><br/>
|
||||
<h4>Move two Stars and three Ants to most left target
|
||||
and one Label3 and four Label2 to most right target.</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" target_outline="true" one_per_target="false">
|
||||
<draggable id="1" label="Label 1" can_reuse="true" />
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg" can_reuse="true" />
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" can_reuse="true" />
|
||||
<draggable id="5" label="Label2" can_reuse="true" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" can_reuse="true" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" can_reuse="true" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" can_reuse="true" />
|
||||
<draggable id="7" label="Label3" can_reuse="true" />
|
||||
|
||||
<target id="t1" x="20" y="20" w="90" h="90"/>
|
||||
<target id="t2" x="300" y="100" w="90" h="90"/>
|
||||
<target id="t3" x="150" y="40" w="50" h="50"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['name4'],
|
||||
'targets': [
|
||||
't1', 't1'
|
||||
],
|
||||
'rule': 'exact'
|
||||
},
|
||||
{
|
||||
'draggables': ['name_with_icon'],
|
||||
'targets': [
|
||||
't1', 't1', 't1'
|
||||
],
|
||||
'rule': 'exact'
|
||||
},
|
||||
{
|
||||
'draggables': ['5'],
|
||||
'targets': [
|
||||
't2', 't2', 't2', 't2'
|
||||
],
|
||||
'rule': 'exact'
|
||||
},
|
||||
{
|
||||
'draggables': ['7'],
|
||||
'targets': [
|
||||
't2'
|
||||
],
|
||||
'rule': 'exact'
|
||||
}
|
||||
]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Draggables can be placed anywhere on base image]</h4><br/>
|
||||
<h4>
|
||||
Place -Grass- in the middle of the image and -Ant- in the
|
||||
right upper corner.</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" >
|
||||
<draggable id="1" label="Label 1" can_reuse="true" />
|
||||
<draggable id="ant" label="Ant" icon="/static/images/images_list/ant.jpg" can_reuse="true" />
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" can_reuse="true" />
|
||||
<draggable id="5" label="Label2" can_reuse="true" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" can_reuse="true" />
|
||||
<draggable id="grass" label="Grass" icon="/static/images/images_list/grass.jpg" can_reuse="true" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" can_reuse="true" />
|
||||
<draggable id="7" label="Label3" can_reuse="true" />
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = {
|
||||
'grass': [[300, 200], 200],
|
||||
'ant': [[500, 0], 200]
|
||||
}
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Another anyof example]</h4><br/>
|
||||
<h4>Please identify the Carbon and Oxygen atoms in the molecule.</h4><br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/ethglycol.jpg" target_outline="true" one_per_target="true">
|
||||
<draggable id="l1_c" label="Carbon" can_reuse="true" />
|
||||
<draggable id="l2" label="Methane" can_reuse="true" />
|
||||
<draggable id="l3_o" label="Oxygen" can_reuse="true" />
|
||||
<draggable id="l4" label="Calcium" can_reuse="true" />
|
||||
<draggable id="l7" label="Hydrogen" can_reuse="true" />
|
||||
|
||||
<target id="t1_o" x="10" y="67" w="100" h="100"/>
|
||||
<target id="t2" x="133" y="3" w="70" h="70"/>
|
||||
<target id="t3" x="2" y="384" w="70" h="70"/>
|
||||
<target id="t4" x="95" y="386" w="70" h="70"/>
|
||||
<target id="t5_c" x="94" y="293" w="91" h="91"/>
|
||||
<target id="t6_c" x="328" y="294" w="91" h="91"/>
|
||||
<target id="t7" x="393" y="463" w="70" h="70"/>
|
||||
<target id="t8" x="344" y="214" w="70" h="70"/>
|
||||
<target id="t9_o" x="445" y="162" w="100" h="100"/>
|
||||
<target id="t10" x="591" y="132" w="70" h="70"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['l3_o'],
|
||||
'targets': ['t1_o', 't9_o'],
|
||||
'rule': 'exact'
|
||||
},
|
||||
{
|
||||
'draggables': ['l1_c'],
|
||||
'targets': ['t5_c', 't6_c'],
|
||||
'rule': 'exact'
|
||||
}
|
||||
]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Exact number of draggables for a set of targets.]</h4><br/>
|
||||
<h4>Drag two Grass and one Star to first or second positions, and three Cloud to any of the three positions.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" target_outline="true" one_per_target="false">
|
||||
<draggable id="1" label="Label 1" can_reuse="true" />
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg" can_reuse="true" />
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" can_reuse="true" />
|
||||
<draggable id="5" label="Label2" can_reuse="true" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" can_reuse="true" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" can_reuse="true" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" can_reuse="true" />
|
||||
<draggable id="7" label="Label3" can_reuse="true" />
|
||||
|
||||
<target id="t1" x="20" y="20" w="90" h="90"/>
|
||||
<target id="t2" x="300" y="100" w="90" h="90"/>
|
||||
<target id="t3" x="150" y="40" w="50" h="50"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['name_label_icon3', 'name_label_icon3'],
|
||||
'targets': ['t1', 't3'],
|
||||
'rule': 'unordered_equal+number'
|
||||
},
|
||||
{
|
||||
'draggables': ['name4'],
|
||||
'targets': ['t1', 't3'],
|
||||
'rule': 'anyof+number'
|
||||
},
|
||||
{
|
||||
'draggables': ['with_icon', 'with_icon', 'with_icon'],
|
||||
'targets': ['t1', 't2', 't3'],
|
||||
'rule': 'anyof+number'
|
||||
}
|
||||
]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[As many as you like draggables for a set of targets.]</h4><br/>
|
||||
<h4>Drag some Grass to any of the targets, and some Stars to either first or last target.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/cow.png" target_outline="true" one_per_target="false">
|
||||
<draggable id="1" label="Label 1" can_reuse="true" />
|
||||
<draggable id="name_with_icon" label="Ant" icon="/static/images/images_list/ant.jpg" can_reuse="true" />
|
||||
<draggable id="with_icon" label="Cloud" icon="/static/images/images_list/cloud.jpg" can_reuse="true" />
|
||||
<draggable id="5" label="Label2" can_reuse="true" />
|
||||
<draggable id="2" label="Drop" icon="/static/images/images_list/drop.jpg" can_reuse="true" />
|
||||
<draggable id="name_label_icon3" label="Grass" icon="/static/images/images_list/grass.jpg" can_reuse="true" />
|
||||
<draggable id="name4" label="Star" icon="/static/images/images_list/star.png" can_reuse="true" />
|
||||
<draggable id="7" label="Label3" can_reuse="true" />
|
||||
|
||||
<target id="t1" x="20" y="20" w="90" h="90"/>
|
||||
<target id="t2" x="300" y="100" w="90" h="90"/>
|
||||
<target id="t3" x="150" y="40" w="50" h="50"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['name_label_icon3'],
|
||||
'targets': ['t1', 't2', 't3'],
|
||||
'rule': 'anyof'
|
||||
},
|
||||
{
|
||||
'draggables': ['name4'],
|
||||
'targets': ['t1', 't2'],
|
||||
'rule': 'anyof'
|
||||
}
|
||||
]
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
</problem>
|
||||
@@ -0,0 +1,262 @@
|
||||
<problem display_name="Drag and drop demos chem features: drag and drop icons or labels
|
||||
to proper positions." attempts="10">
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Simple grading example: draggables on draggables]</h4><br/>
|
||||
<h4>Describe carbon molecule in LCAO-MO.</h4><br/>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/lcao-mo/lcao-mo.jpg" target_outline="true" >
|
||||
|
||||
<!-- filled bond -->
|
||||
<draggable id="up_and_down" icon="/static/images/images_list/lcao-mo/u_d.png" can_reuse="true" />
|
||||
<!-- up bond -->
|
||||
<draggable id="up" icon="/static/images/images_list/lcao-mo/up.png" can_reuse="true" />
|
||||
|
||||
<draggable id="s" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="s orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p" icon="/static/images/images_list/lcao-mo/orbital_triple.png" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
<target id="2" x="34" y="0" w="32" h="32"/>
|
||||
<target id="3" x="68" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<!-- positions of electrons and electron pairs -->
|
||||
<target id="s_l" x="130" y="360" w="32" h="32"/>
|
||||
<target id="s_r" x="505" y="360" w="32" h="32"/>
|
||||
<target id="p_l" x="80" y="100" w="100" h="32"/>
|
||||
<target id="p_r" x="465" y="100" w="100" h="32"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['p'],
|
||||
'targets': ['p_l', 'p_r'],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['s'],
|
||||
'targets': ['s_l', 's_r'],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['up_and_down'],
|
||||
'targets': [
|
||||
's_l[s][1]', 's_r[s][1]'
|
||||
],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['up'],
|
||||
'targets': [
|
||||
'p_l[p][1]', 'p_l[p][3]', 'p_r[p][1]', 'p_r[p][3]'
|
||||
],
|
||||
'rule': 'unordered_equal'
|
||||
}
|
||||
]
|
||||
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Complex grading example: draggables on draggables]</h4><br/>
|
||||
<h4>Describe carbon molecule in LCAO-MO.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/lcao-mo/lcao-mo-clean.jpg" target_outline="true" >
|
||||
|
||||
<!-- filled bond -->
|
||||
<draggable id="up_and_down" icon="/static/images/images_list/lcao-mo/u_d.png" can_reuse="true" />
|
||||
<!-- up bond -->
|
||||
<draggable id="up" icon="/static/images/images_list/lcao-mo/up.png" can_reuse="true" />
|
||||
|
||||
<!-- images that should not be dragged -->
|
||||
<draggable id="down" icon="/static/images/images_list/lcao-mo/d.png" can_reuse="true" />
|
||||
|
||||
<draggable id="s" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="s orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p" icon="/static/images/images_list/lcao-mo/orbital_triple.png" can_reuse="true" label="p orbital" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
<target id="2" x="34" y="0" w="32" h="32"/>
|
||||
<target id="3" x="68" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="s-sigma" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="s-sigma orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="s-sigma*" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="s-sigma* orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p-pi" icon="/static/images/images_list/lcao-mo/orbital_double.png" label="p-pi orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
<target id="2" x="34" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p-sigma" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="p-sigma orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p-pi*" icon="/static/images/images_list/lcao-mo/orbital_double.png" label="p-pi* orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
<target id="2" x="34" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p-sigma*" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="p-sigma* orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<!-- positions of electrons and electron pairs -->
|
||||
<target id="s-left-target" x="130" y="360" w="32" h="32"/>
|
||||
<target id="s-right-target" x="505" y="360" w="32" h="32"/>
|
||||
<target id="s-sigma-target" x="315" y="425" w="32" h="32"/>
|
||||
<target id="s-sigma*-target" x="315" y="290" w="32" h="32"/>
|
||||
<target id="p-left-target" x="80" y="100" w="100" h="32"/>
|
||||
<target id="p-right-target" x="480" y="100" w="100" h="32"/>
|
||||
<target id="p-pi-target" x="300" y="220" w="66" h="32"/>
|
||||
<target id="p-sigma-target" x="315" y="170" w="32" h="32"/>
|
||||
<target id="p-pi*-target" x="300" y="40" w="66" h="32"/>
|
||||
<target id="p-sigma*-target" x="315" y="0" w="32" h="32"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{'draggables': ['p'], 'targets': ['p-left-target', 'p-right-target'], 'rule': 'unordered_equal'},
|
||||
{'draggables': ['s'], 'targets': ['s-left-target', 's-right-target'], 'rule': 'unordered_equal'},
|
||||
{'draggables': ['s-sigma'], 'targets': ['s-sigma-target'], 'rule': 'exact'},
|
||||
{'draggables': ['s-sigma*'], 'targets': ['s-sigma*-target'], 'rule': 'exact'},
|
||||
{'draggables': ['p-pi'], 'targets': ['p-pi-target'], 'rule': 'exact'},
|
||||
{'draggables': ['p-sigma'], 'targets': ['p-sigma-target'], 'rule': 'exact'},
|
||||
{'draggables': ['p-pi*'], 'targets': ['p-pi*-target'], 'rule': 'exact'},
|
||||
{'draggables': ['p-sigma*'], 'targets': ['p-sigma*-target'], 'rule': 'exact'},
|
||||
{
|
||||
'draggables': ['up_and_down'],
|
||||
'targets': ['s-left-target[s][1]', 's-right-target[s][1]', 's-sigma-target[s-sigma][1]', 's-sigma*-target[s-sigma*][1]', 'p-pi-target[p-pi][1]', 'p-pi-target[p-pi][2]'],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['up'],
|
||||
'targets': ['p-left-target[p][1]', 'p-left-target[p][2]', 'p-right-target[p][2]', 'p-right-target[p][3]',],
|
||||
'rule': 'unordered_equal'
|
||||
}
|
||||
]
|
||||
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
<customresponse>
|
||||
<text>
|
||||
<h4>[Complex grading example: no draggables on draggables]</h4><br/>
|
||||
<h4>Describe carbon molecule in LCAO-MO.</h4>
|
||||
<br/>
|
||||
</text>
|
||||
|
||||
<drag_and_drop_input img="/static/images/images_list/lcao-mo/lcao-mo.jpg" target_outline="true">
|
||||
|
||||
<!-- filled bond -->
|
||||
<draggable id="1" icon="/static/images/images_list/lcao-mo/u_d.png" can_reuse="true" />
|
||||
|
||||
<!-- up bond -->
|
||||
<draggable id="7" icon="/static/images/images_list/lcao-mo/up.png" can_reuse="true" />
|
||||
|
||||
<!-- sigma -->
|
||||
<draggable id="11" icon="/static/images/images_list/lcao-mo/sigma.png" can_reuse="true" />
|
||||
|
||||
<!-- sigma* -->
|
||||
<draggable id="13" icon="/static/images/images_list/lcao-mo/sigma_s.png" can_reuse="true" />
|
||||
|
||||
<!-- pi -->
|
||||
<draggable id="15" icon="/static/images/images_list/lcao-mo/pi.png" can_reuse="true" />
|
||||
|
||||
<!-- pi* -->
|
||||
<draggable id="16" icon="/static/images/images_list/lcao-mo/pi_s.png" can_reuse="true" />
|
||||
|
||||
<!-- images that should not be dragged -->
|
||||
<draggable id="17" icon="/static/images/images_list/lcao-mo/d.png" can_reuse="true" />
|
||||
|
||||
<!-- positions of electrons and electron pairs -->
|
||||
<target id="s_left" x="130" y="360" w="32" h="32"/>
|
||||
<target id="s_right" x="505" y="360" w="32" h="32"/>
|
||||
<target id="s_sigma" x="320" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star" x="320" y="290" w="32" h="32"/>
|
||||
<target id="p_left_1" x="80" y="100" w="32" h="32"/>
|
||||
<target id="p_left_2" x="125" y="100" w="32" h="32"/>
|
||||
<target id="p_left_3" x="175" y="100" w="32" h="32"/>
|
||||
<target id="p_right_1" x="465" y="100" w="32" h="32"/>
|
||||
<target id="p_right_2" x="515" y="100" w="32" h="32"/>
|
||||
<target id="p_right_3" x="560" y="100" w="32" h="32"/>
|
||||
<target id="p_pi_1" x="290" y="220" w="32" h="32"/>
|
||||
<target id="p_pi_2" x="335" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma" x="315" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_1" x="290" y="40" w="32" h="32"/>
|
||||
<target id="p_pi_star_2" x="340" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star" x="315" y="0" w="32" h="32"/>
|
||||
|
||||
<!-- positions of names of energy levels -->
|
||||
<target id="s_sigma_name" x="400" y="425" w="32" h="32"/>
|
||||
<target id="s_sigma_star_name" x="400" y="290" w="32" h="32"/>
|
||||
<target id="p_pi_name" x="400" y="220" w="32" h="32"/>
|
||||
<target id="p_sigma_name" x="400" y="170" w="32" h="32"/>
|
||||
<target id="p_pi_star_name" x="400" y="40" w="32" h="32"/>
|
||||
<target id="p_sigma_star_name" x="400" y="0" w="32" h="32"/>
|
||||
|
||||
</drag_and_drop_input>
|
||||
|
||||
<answer type="loncapa/python"><![CDATA[
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['1'],
|
||||
'targets': [
|
||||
's_left', 's_right', 's_sigma', 's_sigma_star', 'p_pi_1', 'p_pi_2'
|
||||
],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['7'],
|
||||
'targets': ['p_left_1', 'p_left_2', 'p_right_2','p_right_3'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['11'],
|
||||
'targets': ['s_sigma_name', 'p_sigma_name'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['13'],
|
||||
'targets': ['s_sigma_star_name', 'p_sigma_star_name'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['15'],
|
||||
'targets': ['p_pi_name'],
|
||||
'rule': 'exact'
|
||||
}, {
|
||||
'draggables': ['16'],
|
||||
'targets': ['p_pi_star_name'],
|
||||
'rule': 'exact'
|
||||
}]
|
||||
|
||||
if draganddrop.grade(submission[0], correct_answer):
|
||||
correct = ['correct']
|
||||
else:
|
||||
correct = ['incorrect']
|
||||
]]></answer>
|
||||
</customresponse>
|
||||
|
||||
</problem>
|
||||
@@ -0,0 +1,425 @@
|
||||
**********************************************
|
||||
XML format of drag and drop input [inputtypes]
|
||||
**********************************************
|
||||
|
||||
.. module:: drag_and_drop_input
|
||||
|
||||
Format description
|
||||
==================
|
||||
|
||||
The main tag of Drag and Drop (DnD) input is::
|
||||
|
||||
<drag_and_drop_input> ... </drag_and_drop_input>
|
||||
|
||||
``drag_and_drop_input`` can include any number of the following 2 tags:
|
||||
``draggable`` and ``target``.
|
||||
|
||||
drag_and_drop_input tag
|
||||
-----------------------
|
||||
|
||||
The main container for a single instance of DnD. The following attributes can
|
||||
be specified for this tag::
|
||||
|
||||
img - Relative path to an image that will be the base image. All draggables
|
||||
can be dragged onto it.
|
||||
target_outline - Specify whether an outline (gray dashed line) should be
|
||||
drawn around targets (if they are specified). It can be either
|
||||
'true' or 'false'. If not specified, the default value is
|
||||
'false'.
|
||||
one_per_target - Specify whether to allow more than one draggable to be
|
||||
placed onto a single target. It can be either 'true' or 'false'. If
|
||||
not specified, the default value is 'true'.
|
||||
no_labels - default is false, in default behaviour if label is not set, label
|
||||
is obtained from id. If no_labels is true, labels are not automatically
|
||||
populated from id, and one can not set labels and obtain only icons.
|
||||
|
||||
draggable tag
|
||||
-------------
|
||||
|
||||
Draggable tag specifies a single draggable object which has the following
|
||||
attributes::
|
||||
|
||||
id - Unique identifier of the draggable object.
|
||||
label - Human readable label that will be shown to the user.
|
||||
icon - Relative path to an image that will be shown to the user.
|
||||
can_reuse - true or false, default is false. If true, same draggable can be
|
||||
used multiple times.
|
||||
|
||||
A draggable is what the user must drag out of the slider and place onto the
|
||||
base image. After a drag operation, if the center of the draggable ends up
|
||||
outside the rectangular dimensions of the image, it will be returned back
|
||||
to the slider.
|
||||
|
||||
In order for the grader to work, it is essential that a unique ID
|
||||
is provided. Otherwise, there will be no way to tell which draggable is at what
|
||||
coordinate, or over what target. Label and icon attributes are optional. If
|
||||
they are provided they will be used, otherwise, you can have an empty
|
||||
draggable. The path is relative to 'course_folder' folder, for example,
|
||||
/static/images/img1.png.
|
||||
|
||||
target tag
|
||||
----------
|
||||
|
||||
Target tag specifies a single target object which has the following required
|
||||
attributes::
|
||||
|
||||
id - Unique identifier of the target object.
|
||||
x - X-coordinate on the base image where the top left corner of the target
|
||||
will be positioned.
|
||||
y - Y-coordinate on the base image where the top left corner of the target
|
||||
will be positioned.
|
||||
w - Width of the target.
|
||||
h - Height of the target.
|
||||
|
||||
A target specifies a place on the base image where a draggable can be
|
||||
positioned. By design, if the center of a draggable lies within the target
|
||||
(i.e. in the rectangle defined by [[x, y], [x + w, y + h]], then it is within
|
||||
the target. Otherwise, it is outside.
|
||||
|
||||
If at lest one target is provided, the behavior of the client side logic
|
||||
changes. If a draggable is not dragged on to a target, it is returned back to
|
||||
the slider.
|
||||
|
||||
If no targets are provided, then a draggable can be dragged and placed anywhere
|
||||
on the base image.
|
||||
|
||||
Targets on draggables
|
||||
---------------------
|
||||
|
||||
Sometimes it is not enough to have targets only on the base image, and all of the
|
||||
draggables on these targets. If a complex problem exists where a draggable must
|
||||
become itself a target (or many targets), then the following extended syntax
|
||||
can be used: ::
|
||||
|
||||
<draggable {attribute list}>
|
||||
<target {attribute list} />
|
||||
<target {attribute list} />
|
||||
<target {attribute list} />
|
||||
...
|
||||
</draggable>
|
||||
|
||||
The attribute list in the tags above ('draggable' and 'target') is the same as for
|
||||
normal 'draggable' and 'target' tags. The only difference is when you will be
|
||||
specifying inner target position coordinates. Using the 'x' and 'y' attributes you
|
||||
are setting the offset of the inner target from the upper-left corner of the
|
||||
parent draggable (that contains the inner target).
|
||||
|
||||
Limitations of targets on draggables
|
||||
------------------------------------
|
||||
|
||||
1.) Currently there is a limitation to the level of nesting of targets.
|
||||
|
||||
Even though you can pile up a large number of draggables on targets that themselves
|
||||
are on draggables, the Drag and Drop instance will be graded only in the case if
|
||||
there is a maximum of two levels of targets. The first level are the "base" targets.
|
||||
They are attached to the base image. The second level are the targets defined on
|
||||
draggables.
|
||||
|
||||
2.) Another limitation is that the target bounds are not checked against
|
||||
other targets.
|
||||
|
||||
For now, it is the responsibility of the person who is constructing the course
|
||||
material to make sure that there is no overlapping of targets. It is also preferable
|
||||
that targets on draggables are smaller than the actual parent draggable. Technically
|
||||
this is not necessary, but from the usability perspective it is desirable.
|
||||
|
||||
3.) You can have targets on draggables only in the case when there are base targets
|
||||
defined (base targets are attached to the base image).
|
||||
|
||||
If you do not have base targets, then you can only have a single level of nesting
|
||||
(draggables on the base image). In this case the client side will be reporting (x,y)
|
||||
positions of each draggables on the base image.
|
||||
|
||||
Correct answer format
|
||||
---------------------
|
||||
|
||||
(NOTE: For specifying answers for targets on draggables please see next section.)
|
||||
|
||||
There are two correct answer formats: short and long
|
||||
If short from correct answer is mapping of 'draggable_id' to 'target_id'::
|
||||
|
||||
correct_answer = {'grass': [[300, 200], 200], 'ant': [[500, 0], 200]}
|
||||
correct_answer = {'name4': 't1', '7': 't2'}
|
||||
|
||||
In long form correct answer is list of dicts. Every dict has 3 keys:
|
||||
draggables, targets and rule. For example::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['7', '8'],
|
||||
'targets': ['t5_c', 't6_c'],
|
||||
'rule': 'anyof'
|
||||
},
|
||||
{
|
||||
'draggables': ['1', '2'],
|
||||
'targets': ['t2_h', 't3_h', 't4_h', 't7_h', 't8_h', 't10_h'],
|
||||
'rule': 'anyof'
|
||||
}]
|
||||
|
||||
Draggables is list of draggables id. Target is list of targets id, draggables
|
||||
must be dragged to with considering rule. Rule is string.
|
||||
|
||||
Draggables in dicts inside correct_answer list must not intersect!!!
|
||||
|
||||
Wrong (for draggable id 7)::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['7', '8'],
|
||||
'targets': ['t5_c', 't6_c'],
|
||||
'rule': 'anyof'
|
||||
},
|
||||
{
|
||||
'draggables': ['7', '2'],
|
||||
'targets': ['t2_h', 't3_h', 't4_h', 't7_h', 't8_h', 't10_h'],
|
||||
'rule': 'anyof'
|
||||
}]
|
||||
|
||||
Rules are: exact, anyof, unordered_equal, anyof+number, unordered_equal+number
|
||||
|
||||
|
||||
.. such long lines are needed for sphinx to display lists correctly
|
||||
|
||||
- Exact rule means that targets for draggable id's in user_answer are the same that targets from correct answer. For example, for draggables 7 and 8 user must drag 7 to target1 and 8 to target2 if correct_answer is::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['7', '8'],
|
||||
'targets': ['tartget1', 'target2'],
|
||||
'rule': 'exact'
|
||||
}]
|
||||
|
||||
|
||||
- unordered_equal rule allows draggables be dragged to targets unordered. If one want to allow for student to drag 7 to target1 or target2 and 8 to target2 or target 1 and 7 and 8 must be in different targets, then correct answer must be::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['7', '8'],
|
||||
'targets': ['tartget1', 'target2'],
|
||||
'rule': 'unordered_equal'
|
||||
}]
|
||||
|
||||
|
||||
- Anyof rule allows draggables to be dragged to any of targets. If one want to allow for student to drag 7 and 8 to target1 or target2, which means that if 7 is on target1 and 8 is on target1 or 7 on target2 and 8 on target2 or 7 on target1 and 8 on target2. Any of theese are correct which anyof rule::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['7', '8'],
|
||||
'targets': ['tartget1', 'target2'],
|
||||
'rule': 'anyof'
|
||||
}]
|
||||
|
||||
|
||||
- If you have can_reuse true, then you, for example, have draggables a,b,c and 10 targets. These will allow you to drag 4 'a' draggables to ['target1', 'target4', 'target7', 'target10'] , you do not need to write 'a' four times. Also this will allow you to drag 'b' draggable to target2 or target5 for target5 and target2 etc..::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['a'],
|
||||
'targets': ['target1', 'target4', 'target7', 'target10'],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['b'],
|
||||
'targets': ['target2', 'target5', 'target8'],
|
||||
'rule': 'anyof'
|
||||
},
|
||||
{
|
||||
'draggables': ['c'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'unordered_equal'
|
||||
}]
|
||||
|
||||
- And sometimes you want to allow drag only two 'b' draggables, in these case you should use 'anyof+number' of 'unordered_equal+number' rule::
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['a', 'a', 'a'],
|
||||
'targets': ['target1', 'target4', 'target7'],
|
||||
'rule': 'unordered_equal+numbers'
|
||||
},
|
||||
{
|
||||
'draggables': ['b', 'b'],
|
||||
'targets': ['target2', 'target5', 'target8'],
|
||||
'rule': 'anyof+numbers'
|
||||
},
|
||||
{
|
||||
'draggables': ['c'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'unordered_equal'
|
||||
}]
|
||||
|
||||
In case if we have no multiple draggables per targets (one_per_target="true"),
|
||||
for same number of draggables, anyof is equal to unordered_equal
|
||||
|
||||
If we have can_reuse=true, than one must use only long form of correct answer.
|
||||
|
||||
Answer format for targets on draggables
|
||||
---------------------------------------
|
||||
|
||||
As with the cases described above, an answer must provide precise positioning for
|
||||
each draggable (on which targets it must reside). In the case when a draggable must
|
||||
be placed on a target that itself is on a draggable, then the answer must contain
|
||||
the chain of target-draggable-target. It is best to understand this on an example.
|
||||
|
||||
Suppose we have three draggables - 'up', 's', and 'p'. Draggables 's', and 'p' have targets
|
||||
on themselves. More specifically, 'p' has three targets - '1', '2', and '3'. The first
|
||||
requirement is that 's', and 'p' are positioned on specific targets on the base image.
|
||||
The second requirement is that draggable 'up' is positioned on specific targets of
|
||||
draggable 'p'. Below is an excerpt from a problem.::
|
||||
|
||||
<draggable id="up" icon="/static/images/images_list/lcao-mo/up.png" can_reuse="true" />
|
||||
|
||||
<draggable id="s" icon="/static/images/images_list/lcao-mo/orbital_single.png" label="s orbital" can_reuse="true" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
<draggable id="p" icon="/static/images/images_list/lcao-mo/orbital_triple.png" can_reuse="true" label="p orbital" >
|
||||
<target id="1" x="0" y="0" w="32" h="32"/>
|
||||
<target id="2" x="34" y="0" w="32" h="32"/>
|
||||
<target id="3" x="68" y="0" w="32" h="32"/>
|
||||
</draggable>
|
||||
|
||||
...
|
||||
|
||||
correct_answer = [
|
||||
{
|
||||
'draggables': ['p'],
|
||||
'targets': ['p-left-target', 'p-right-target'],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['s'],
|
||||
'targets': ['s-left-target', 's-right-target'],
|
||||
'rule': 'unordered_equal'
|
||||
},
|
||||
{
|
||||
'draggables': ['up'],
|
||||
'targets': ['p-left-target[p][1]', 'p-left-target[p][2]', 'p-right-target[p][2]', 'p-right-target[p][3]',],
|
||||
'rule': 'unordered_equal'
|
||||
}
|
||||
]
|
||||
|
||||
Note that it is a requirement to specify rules for all draggables, even if some draggable gets included
|
||||
in more than one chain.
|
||||
|
||||
Grading logic
|
||||
-------------
|
||||
|
||||
1. User answer (that comes from browser) and correct answer (from xml) are parsed to the same format::
|
||||
|
||||
group_id: group_draggables, group_targets, group_rule
|
||||
|
||||
|
||||
Group_id is ordinal number, for every dict in correct answer incremental
|
||||
group_id is assigned: 0, 1, 2, ...
|
||||
|
||||
Draggables from user answer are added to same group_id where identical draggables
|
||||
from correct answer are, for example::
|
||||
|
||||
If correct_draggables[group_0] = [t1, t2] then
|
||||
user_draggables[group_0] are all draggables t1 and t2 from user answer:
|
||||
[t1] or [t1, t2] or [t1, t2, t2] etc..
|
||||
|
||||
2. For every group from user answer, for that group draggables, if 'number' is in group rule, set() is applied,
|
||||
if 'number' is not in rule, set is not applied::
|
||||
|
||||
set() : [t1, t2, t3, t3] -> [t1, t2, ,t3]
|
||||
|
||||
For every group, at this step, draggables lists are equal.
|
||||
|
||||
|
||||
3. For every group, lists of targets are compared using rule for that group.
|
||||
|
||||
|
||||
Set and '+number' cases
|
||||
.......................
|
||||
|
||||
Set() and '+number' are needed only for case of reusable draggables,
|
||||
for other cases there are no equal draggables in list, so set() does nothing.
|
||||
|
||||
.. such long lines needed for sphinx to display nicely
|
||||
|
||||
* Usage of set() operation allows easily create rule for case of "any number of same draggable can be dragged to some targets"::
|
||||
|
||||
{
|
||||
'draggables': ['draggable_1'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'anyof'
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
* 'number' rule is used for the case of reusable draggables, when one want to fix number of draggable to drag. In this example only two instances of draggables_1 are allowed to be dragged::
|
||||
|
||||
{
|
||||
'draggables': ['draggable_1', 'draggable_1'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'anyof+number'
|
||||
}
|
||||
|
||||
|
||||
* Note, that in using rule 'exact', one does not need 'number', because you can't recognize from user interface which reusable draggable is on which target. Absurd example::
|
||||
|
||||
{
|
||||
'draggables': ['draggable_1', 'draggable_1', 'draggable_2'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'exact'
|
||||
}
|
||||
|
||||
|
||||
Correct handling of this example is to create different rules for draggable_1 and
|
||||
draggable_2
|
||||
|
||||
* For 'unordered_equal' (or 'exact' too) we don't need 'number' if you have only same draggable in group, as targets length will provide constraint for the number of draggables::
|
||||
|
||||
{
|
||||
'draggables': ['draggable_1'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'unordered_equal'
|
||||
}
|
||||
|
||||
|
||||
This means that only three draggaggables 'draggable_1' can be dragged.
|
||||
|
||||
* But if you have more that one different reusable draggable in list, you may use 'number' rule::
|
||||
|
||||
{
|
||||
'draggables': ['draggable_1', 'draggable_1', 'draggable_2'],
|
||||
'targets': ['target3', 'target6', 'target9'],
|
||||
'rule': 'unordered_equal+number'
|
||||
}
|
||||
|
||||
|
||||
If not use number, draggables list will be setted to ['draggable_1', 'draggable_2']
|
||||
|
||||
|
||||
|
||||
|
||||
Logic flow
|
||||
----------
|
||||
|
||||
(Click on image to see full size version.)
|
||||
|
||||
.. image:: draganddrop_logic_flow.png
|
||||
:width: 100%
|
||||
:target: _images/draganddrop_logic_flow.png
|
||||
|
||||
|
||||
Example
|
||||
=======
|
||||
|
||||
Examples of draggables that can't be reused
|
||||
-------------------------------------------
|
||||
|
||||
.. literalinclude:: drag-n-drop-demo.xml
|
||||
|
||||
Draggables can be reused
|
||||
------------------------
|
||||
|
||||
.. literalinclude:: drag-n-drop-demo2.xml
|
||||
|
||||
Examples of targets on draggables
|
||||
---------------------------------
|
||||
|
||||
.. literalinclude:: drag-n-drop-demo3.xml
|
||||
Binary file not shown.
|
After Width: | Height: | Size: 298 KiB |
171
docs/data/source/course_data_formats/grading.rst
Normal file
171
docs/data/source/course_data_formats/grading.rst
Normal file
@@ -0,0 +1,171 @@
|
||||
##############
|
||||
Course Grading
|
||||
##############
|
||||
This document is written to help professors understand how a final grade for a
|
||||
course is computed.
|
||||
|
||||
Course grading is the process of taking all of the problems scores for a student
|
||||
in a course and generating a final score (and corresponding letter grade). This
|
||||
grading process can be split into two phases - totaling sections and section
|
||||
weighting.
|
||||
|
||||
*****************
|
||||
Totaling sections
|
||||
*****************
|
||||
The process of totaling sections is to get a percentage score (between 0.0 and
|
||||
1.0) for every section in the course. A section is any module that is a direct
|
||||
child of a chapter. For example, psets, labs, and sequences are all common
|
||||
sections. Only the *percentage* on the section will be available to compute the
|
||||
final grade, *not* the final number of points earned / possible.
|
||||
|
||||
.. important::
|
||||
For a section to be included in the final grade, the policies file must set
|
||||
`graded = True` for the section.
|
||||
|
||||
For each section, the grading function retrieves all problems within the
|
||||
section. The section percentage is computed as (total points earned) / (total
|
||||
points possible).
|
||||
|
||||
******************
|
||||
Weighting Problems
|
||||
******************
|
||||
In some cases, one might want to give weights to problems within a section. For
|
||||
example, a final exam might contain four questions each worth 1 point by default.
|
||||
This means each question would by default have the same weight. If one wanted
|
||||
the first problem to be worth 50% of the final exam, the policy file could specify
|
||||
weights of 30, 10, 10, and 10 to the four problems, respectively.
|
||||
|
||||
Note that the default weight of a problem **is not 1**. The default weight of a
|
||||
problem is the module's `max_grade`.
|
||||
|
||||
If weighting is set, each problem is worth the number of points assigned, regardless of the number of responses it contains.
|
||||
|
||||
Consider a Homework section that contains two problems.
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<problem display_name=”Problem 1”>
|
||||
<numericalresponse> ... </numericalreponse>
|
||||
</problem>
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<problem display_name=”Problem 2”>
|
||||
<numericalresponse> ... </numericalreponse>
|
||||
<numericalresponse> ... </numericalreponse>
|
||||
<numericalresponse> ... </numericalreponse>
|
||||
</problem>
|
||||
|
||||
Without weighting, Problem 1 is worth 25% of the assignment, and Problem 2 is worth 75% of the assignment.
|
||||
|
||||
Weighting for the problems can be set in the policy.json file.
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
"problem/problem1": {
|
||||
"weight": 2
|
||||
},
|
||||
"problem/problem2": {
|
||||
"weight": 2
|
||||
},
|
||||
|
||||
With the above weighting, Problems 1 and 2 are each worth 50% of the assignment.
|
||||
|
||||
Please note: When problems have weight, the point value is automatically included in the display name *except* when `"weight": 1`. When the weight is 1, no visual change occurs in the display name, leaving the point value open to interpretation to the student.
|
||||
|
||||
|
||||
******************
|
||||
Weighting Sections
|
||||
******************
|
||||
Once each section has a percentage score, we must total those sections into a
|
||||
final grade. Of course, not every section has equal weight in the final grade.
|
||||
The policies for weighting sections into a final grade are specified in the
|
||||
grading_policy.json file.
|
||||
|
||||
The `grading_policy.json` file specifies several sub-graders that are each given
|
||||
a weight and factored into the final grade. There are currently two types of
|
||||
sub-graders, section format graders and single section graders.
|
||||
|
||||
We will use this simple example of a grader with one section format grader and
|
||||
one single section grader.
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
"GRADER" : [
|
||||
{
|
||||
"type" : "Homework",
|
||||
"min_count" : 12,
|
||||
"drop_count" : 2,
|
||||
"short_label" : "HW",
|
||||
"weight" : 0.4
|
||||
},
|
||||
{
|
||||
"type" : "Final",
|
||||
"name" : "Final Exam",
|
||||
"short_label" : "Final",
|
||||
"weight" : 0.6
|
||||
}
|
||||
]
|
||||
|
||||
Section Format Graders
|
||||
======================
|
||||
A section format grader grades a set of sections with the same format, as
|
||||
defined in the course policy file. To make a vertical named Homework1 be graded
|
||||
by the Homework section format grader, the following definition would be in the
|
||||
course policy file.
|
||||
|
||||
.. code-block:: json
|
||||
|
||||
"vertical/Homework1": {
|
||||
"display_name": "Homework 1",
|
||||
"graded": true,
|
||||
"format": "Homework"
|
||||
},
|
||||
|
||||
|
||||
In the example above, the section format grader declares that it will expect to
|
||||
find at least 12 sections with the format "Homework". It will drop the lowest 2.
|
||||
All of the homework assignments will have equal weight, relative to each other
|
||||
(except, of course, for the assignments that are dropped).
|
||||
|
||||
This format supports forecasting the number of homework assignments. For
|
||||
example, if the course only has 3 homeworks written, but the section format
|
||||
grader has been told to expect 12, the missing 9 will have an assumed 0% and
|
||||
will still show up in the grade breakdown.
|
||||
|
||||
A section format grader will also show the average of that section in the grade
|
||||
breakdown (shown on the Progress page, gradebook, etc.).
|
||||
|
||||
|
||||
Single Section Graders
|
||||
======================
|
||||
A single section grader grades exactly that - a single section. If a section
|
||||
is found with a matching format and display name then the score of that section
|
||||
is used. If not, a score of 0% is assumed.
|
||||
|
||||
|
||||
Combining sub-graders
|
||||
=====================
|
||||
The final grade is computed by taking the score and weight of each sub grader.
|
||||
In the above example, homework will be 40% of the final grade. The final exam
|
||||
will be 60% of the final grade.
|
||||
|
||||
**************************
|
||||
Displaying the final grade
|
||||
**************************
|
||||
The final grade is then rounded up to the nearest percentage point. This is so
|
||||
the system can consistently display a percentage without worrying whether the
|
||||
displayed percentage has been rounded up or down (potentially misleading the
|
||||
student). The formula for the rounding is::
|
||||
|
||||
rounded_percent = round(computed_percent * 100 + 0.05) / 100
|
||||
|
||||
The grading policy file also specifies the cutoffs for the grade levels. A
|
||||
grade is either A, B, or C. If the student does not reach the cutoff threshold
|
||||
for a C grade then the student has not earned a grade and will not be eligible
|
||||
for a certificate. Letter grades are only awarded to students who have
|
||||
completed the course. There is no notion of a failing letter grade.
|
||||
|
||||
|
||||
|
||||
|
||||
@@ -0,0 +1,563 @@
|
||||
*********************************************
|
||||
XML format of graphical slider tool [xmodule]
|
||||
*********************************************
|
||||
|
||||
.. module:: xml_format_gst
|
||||
|
||||
|
||||
Format description
|
||||
==================
|
||||
|
||||
Graphical slider tool (GST) main tag is::
|
||||
|
||||
<graphical_slider_tool> BODY </graphical_slider_tool>
|
||||
|
||||
``graphical_slider_tool`` tag must have two children tags: ``render``
|
||||
and ``configuration``.
|
||||
|
||||
|
||||
Render tag
|
||||
----------
|
||||
|
||||
Render tag can contain usual html tags mixed with some GST specific tags::
|
||||
|
||||
<slider/> - represents jQuery slider for changing a parameter's value
|
||||
<textbox/> - represents a text input field for changing a parameter's value
|
||||
<plot/> - represents Flot JS plot element
|
||||
|
||||
Also GST will track all elements inside ``<render></render>`` where ``id``
|
||||
attribute is set, and a corresponding parameter referencing that ``id`` is present
|
||||
in the configuration section below. These will be referred to as dynamic elements.
|
||||
|
||||
The contents of the <render> section will be shown to the user after
|
||||
all occurrences of::
|
||||
|
||||
<slider var="{parameter name}" [style="{CSS statements}"] />
|
||||
<textbox var="{parameter name}" [style="{CSS statements}"] />
|
||||
<plot [style="{CSS statements}"] />
|
||||
|
||||
have been converted to actual sliders, text inputs, and a plot graph.
|
||||
Everything in square brackets is optional. After initialization, all
|
||||
text input fields, sliders, and dynamic elements will be set to the initial
|
||||
values of the parameters that they are assigned to.
|
||||
|
||||
``{parameter name}`` specifies the parameter to which the slider or text
|
||||
input will be attached to.
|
||||
|
||||
[style="{CSS statements}"] specifies valid CSS styling. It will be passed
|
||||
directly to the browser without any parsing.
|
||||
|
||||
There is a one-to-one relationship between a slider and a parameter.
|
||||
I.e. for one parameter you can put only one ``<slider>`` in the
|
||||
``<render>`` section. However, you don't have to specify a slider - they
|
||||
are optional.
|
||||
|
||||
There is a many-to-one relationship between text inputs and a
|
||||
parameter. I.e. for one parameter you can put many '<textbox>' elements in
|
||||
the ``<render>`` section. However, you don't have to specify a text
|
||||
input - they are optional.
|
||||
|
||||
You can put only one ``<plot>`` in the ``<render>`` section. It is not
|
||||
required.
|
||||
|
||||
|
||||
Slider tag
|
||||
..........
|
||||
|
||||
Slider tag must have ``var`` attribute and optional ``style`` attribute::
|
||||
|
||||
<slider var='a' style="width:400px;float:left;" />
|
||||
|
||||
After processing, slider tags will be replaced by jQuery UI sliders with applied
|
||||
``style`` attribute.
|
||||
|
||||
``var`` attribute must correspond to a parameter. Parameters can be used in any
|
||||
of the ``function`` tags in ``functions`` tag. By moving slider, value of
|
||||
parameter ``a`` will change, and so result of function, that depends on parameter
|
||||
``a``, will also change.
|
||||
|
||||
|
||||
Textbox tag
|
||||
...........
|
||||
|
||||
Texbox tag must have ``var`` attribute and optional ``style`` attribute::
|
||||
|
||||
<textbox var="b" style="width:50px; float:left; margin-left:10px;" />
|
||||
|
||||
After processing, textbox tags will be replaced by html text inputs with applied
|
||||
``style`` attribute. If you want a readonly text input, then you should use a
|
||||
dynamic element instead (see section below "HTML tagsd with ID").
|
||||
|
||||
``var`` attribute must correspond to a parameter. Parameters can be used in any
|
||||
of the ``function`` tags in ``functions`` tag. By changing the value on the text input,
|
||||
value of parameter ``a`` will change, and so result of function, that depends on
|
||||
parameter ``a``, will also change.
|
||||
|
||||
|
||||
Plot tag
|
||||
........
|
||||
|
||||
Plot tag may have optional ``style`` attribute::
|
||||
|
||||
<plot style="width:50px; float:left; margin-left:10px;" />
|
||||
|
||||
After processing plot tags will be replaced by Flot JS plot with applied
|
||||
``style`` attribute.
|
||||
|
||||
|
||||
HTML tags with ID (dynamic elements)
|
||||
....................................
|
||||
|
||||
Any HTML tag with ID, e.g. ``<span id="answer_span_1">`` can be used as a
|
||||
place where result of function can be inserted. To insert function result to
|
||||
an element, element ID must be included in ``function`` tag as ``el_id`` attribute
|
||||
and ``output`` value must be ``"element"``::
|
||||
|
||||
<function output="element" el_id="answer_span_1">
|
||||
function add(a, b, precision) {
|
||||
var x = Math.pow(10, precision || 2);
|
||||
return (Math.round(a * x) + Math.round(b * x)) / x;
|
||||
}
|
||||
|
||||
return add(a, b, 5);
|
||||
</function>
|
||||
|
||||
|
||||
Configuration tag
|
||||
-----------------
|
||||
|
||||
The configuration tag contains parameter settings, graph
|
||||
settings, and function definitions which are to be plotted on the
|
||||
graph and that use specified parameters.
|
||||
|
||||
Configuration tag contains two mandatory tag ``functions`` and ``parameters`` and
|
||||
may contain another ``plot`` tag.
|
||||
|
||||
|
||||
Parameters tag
|
||||
..............
|
||||
|
||||
``Parameters`` tag contains ``parameter`` tags. Each ``parameter`` tag must have
|
||||
``var``, ``max``, ``min``, ``step`` and ``initial`` attributes::
|
||||
|
||||
<parameters>
|
||||
<param var="a" min="-10.0" max="10.0" step="0.1" initial="0" />
|
||||
<param var="b" min="-10.0" max="10.0" step="0.1" initial="0" />
|
||||
</parameters>
|
||||
|
||||
``var`` attribute links min, max, step and initial values to parameter name.
|
||||
|
||||
``min`` attribute is the minimal value that a parameter can take. Slider and input
|
||||
values can not go below it.
|
||||
|
||||
``max`` attribute is the maximal value that a parameter can take. Slider and input
|
||||
values can not go over it.
|
||||
|
||||
``step`` attribute is value of slider step. When a slider increase or decreases
|
||||
the specified parameter, it will do so by the amount specified with 'step'
|
||||
|
||||
``initial`` attribute is the initial value that the specified parameter should be
|
||||
set to. Sliders and inputs will initially show this value.
|
||||
|
||||
The parameter's name is specified by the ``var`` property. All occurrences
|
||||
of sliders and/or text inputs that specify a ``var`` property, will be
|
||||
connected to this parameter - i.e. they will reflect the current
|
||||
value of the parameter, and will be updated when the parameter
|
||||
changes.
|
||||
|
||||
If at lest one of these attributes is not set, then the parameter
|
||||
will not be used, slider's and/or text input elements that specify
|
||||
this parameter will not be activated, and the specified functions
|
||||
which use this parameter will not return a numeric value. This means
|
||||
that neglecting to specify at least one of the attributes for some
|
||||
parameter will have the result of the whole GST instance not working
|
||||
properly.
|
||||
|
||||
|
||||
Functions tag
|
||||
.............
|
||||
|
||||
For the GST to do something, you must defined at least one
|
||||
function, which can use any of the specified parameter values. The
|
||||
function expects to take the ``x`` value, do some calculations, and
|
||||
return the ``y`` value. I.e. this is a 2D plot in Cartesian
|
||||
coordinates. This is how the default function is meant to be used for
|
||||
the graph.
|
||||
|
||||
There are other special cases of functions. They are used mainly for
|
||||
outputting to elements, plot labels, or for custom output. Because
|
||||
the return a single value, and that value is meant for a single element,
|
||||
these function are invoked only with the set of all of the parameters.
|
||||
I.e. no ``x`` value is available inside them. They are useful for
|
||||
showing the current value of a parameter, showing complex static
|
||||
formulas where some parameter's value must change, and other useful
|
||||
things.
|
||||
|
||||
The different style of function is specified by the ``output`` attribute.
|
||||
|
||||
Each function must be defined inside ``function`` tag in ``functions`` tag::
|
||||
|
||||
<functions>
|
||||
<function output="element" el_id="answer_span_1">
|
||||
function add(a, b, precision) {
|
||||
var x = Math.pow(10, precision || 2);
|
||||
return (Math.round(a * x) + Math.round(b * x)) / x;
|
||||
}
|
||||
|
||||
return add(a, b, 5);
|
||||
</function>
|
||||
</functions>
|
||||
|
||||
The parameter names (along with their values, as provided from text
|
||||
inputs and/or sliders), will be available inside all defined
|
||||
functions. A defined function body string will be parsed internally
|
||||
by the browser's JavaScript engine and converted to a true JS
|
||||
function.
|
||||
|
||||
The function's parameter list will automatically be created and
|
||||
populated, and will include the ``x`` (when ``output`` is not specified or
|
||||
is set to ``"graph"``), and all of the specified parameter values (from sliders
|
||||
and text inputs). This means that each of the defined functions will have
|
||||
access to all of the parameter values. You don't have to use them, but
|
||||
they will be there.
|
||||
|
||||
Examples::
|
||||
|
||||
<function>
|
||||
return x;
|
||||
</function>
|
||||
|
||||
<function dot="true" label="\(y_2\)">
|
||||
return (x + a) * Math.sin(x * b);
|
||||
</function>
|
||||
|
||||
<function color="green">
|
||||
function helperFunc(c1) {
|
||||
return c1 * c1 - a;
|
||||
}
|
||||
return helperFunc(x + 10 * a * b) + Math.sin(a - x);
|
||||
</function>
|
||||
|
||||
Required parameters::
|
||||
|
||||
function body:
|
||||
|
||||
A string composing a normal JavaScript function
|
||||
except that there is no function declaration
|
||||
(along with parameters), and no closing bracket.
|
||||
|
||||
So if you normally would have written your
|
||||
JavaScript function like this:
|
||||
|
||||
function myFunc(x, a, b) {
|
||||
return x * a + b;
|
||||
}
|
||||
|
||||
here you must specify just the function body
|
||||
(everything that goes between '{' and '}'). So,
|
||||
you would specify the above function like so (the
|
||||
bare-bone minimum):
|
||||
|
||||
<function>return x * a + b;</function>
|
||||
|
||||
VERY IMPORTANT: Because the function will be passed
|
||||
to the browser as a single string, depending on implementation
|
||||
specifics, the end-of-line characters can be stripped. This
|
||||
means that single line JavaScript comments (starting with "//")
|
||||
can lead to the effect that everything after the first such comment
|
||||
will be treated as a comment. Therefore, it is absolutely
|
||||
necessary that such single line comments are not used when
|
||||
defining functions for GST. You can safely use the alternative
|
||||
multiple line JavaScript comments (such comments start with "/*"
|
||||
and end with "*/).
|
||||
|
||||
VERY IMPORTANT: If you have a large function body, and decide to
|
||||
split it into several lines, than you must wrap it in "CDATA" like
|
||||
so:
|
||||
|
||||
<function>
|
||||
<![CDATA[
|
||||
var dNew;
|
||||
|
||||
dNew = 0.3;
|
||||
|
||||
return x * a + b - dNew;
|
||||
]]>
|
||||
</function>
|
||||
|
||||
Optional parameters::
|
||||
|
||||
|
||||
color: Color name ('red', 'green', etc.) or in the form of
|
||||
'#FFFF00'. If not specified, a default color (different
|
||||
one for each graphed function) will be given by Flot JS.
|
||||
line: A string - 'true' or 'false'. Should the data points be
|
||||
connected by a line on the graph? Default is 'true'.
|
||||
dot: A string - 'true' or 'false'. Should points be shown for
|
||||
each data point on the graph? Default is 'false'.
|
||||
bar: A string - 'true' or 'false'. When set to 'true', points
|
||||
will be plotted as bars.
|
||||
label: A string. If provided, will be shown in the legend, along
|
||||
with the color that was used to plot the function.
|
||||
output: 'element', 'none', 'plot_label', or 'graph'. If not defined,
|
||||
function will be plotted (same as setting 'output' to 'graph').
|
||||
If defined, and other than 'graph', function will not be
|
||||
plotted, but it's output will be inserted into the element
|
||||
with ID specified by 'el_id' attribute.
|
||||
el_id: Id of HTML element, defined in '<render>' section. Value of
|
||||
function will be inserted as content of this element.
|
||||
disable_auto_return: By default, if JavaScript function string is written
|
||||
without a "return" statement, the "return" will be
|
||||
prepended to it. Set to "true" to disable this
|
||||
functionality. This is done so that simple functions
|
||||
can be defined in an easy fashion (for example, "a",
|
||||
which will be translated into "return a").
|
||||
update_on: A string - 'change', or 'slide'. Default (if not set) is
|
||||
'slide'. This defines the event on which a given function is
|
||||
called, and its result is inserted into an element. This
|
||||
setting is relevant only when "output" is other than "graph".
|
||||
|
||||
When specifying ``el_id``, it is essential to set "output" to one of
|
||||
element - GST will invoke the function, and the return of it will be
|
||||
inserted into a HTML element with id specified by ``el_id``.
|
||||
none - GST will simply inoke the function. It is left to the instructor
|
||||
who writes the JavaScript function body to update all necesary
|
||||
HTML elements inside the function, before it exits. This is done
|
||||
so that extra steps can be preformed after an HTML element has
|
||||
been updated with a value. Note, that because the return value
|
||||
from this function is not actually used, it will be tempting to
|
||||
omit the "return" statement. However, in this case, the attribute
|
||||
"disable_auto_return" must be set to "true" in order to prevent
|
||||
GST from inserting a "return" statement automatically.
|
||||
plot_label - GST will process all plot labels (which are strings), and
|
||||
will replace the all instances of substrings specified by
|
||||
``el_id`` with the returned value of the function. This is
|
||||
necessary if you want a label in the graph to have some changing
|
||||
number. Because of the nature of Flot JS, it is impossible to
|
||||
achieve the same effect by setting the "output" attribute
|
||||
to "element", and including a HTML element in the label.
|
||||
|
||||
The above values for "output" will tell GST that the function is meant for an
|
||||
HTML element (not for graph), and that it should not get an 'x' parameter (along
|
||||
with some value).
|
||||
|
||||
|
||||
[Note on MathJax and labels]
|
||||
............................
|
||||
|
||||
Independently of this module, will render all TeX code
|
||||
within the ``<render>`` section into nice mathematical formulas. Just
|
||||
remember to wrap it in one of::
|
||||
|
||||
\( and \) - for inline formulas (formulas surrounded by
|
||||
standard text)
|
||||
\[ and \] - if you want the formula to be a separate line
|
||||
|
||||
It is possible to define a label in standard TeX notation. The JS
|
||||
library MathJax will work on these labels also because they are
|
||||
inserted on top of the plot as standard HTML (text within a DIV).
|
||||
|
||||
If the label is dynamic, i.e. it will contain some text (numeric, or other)
|
||||
that has to be updated on a parameter's change, then one can define
|
||||
a special function to handle this. The "output" of such a function must be
|
||||
set to "none", and the JavaScript code inside this function must update the
|
||||
MathJax element by itself. Before exiting, MathJax typeset function should
|
||||
be called so that the new text will be re-rendered by MathJax. For example::
|
||||
|
||||
<render>
|
||||
...
|
||||
<span id="dynamic_mathjax"></span>
|
||||
</render>
|
||||
...
|
||||
<function output="none" el_id="dynamic_mathjax">
|
||||
<![CDATA[
|
||||
var out_text;
|
||||
|
||||
out_text = "\\[\\mathrm{Percent \\space of \\space treated \\space with \\space YSS=\\frac{"
|
||||
+(treated_men*10)+"\\space men *"
|
||||
+(your_m_tx_yss/100)+"\\space prev. +\\space "
|
||||
+((100-treated_men)*10)+"\\space women *"
|
||||
+(your_f_tx_yss/100)+"\\space prev.}"
|
||||
+"{1000\\space total\\space treated\\space patients}"
|
||||
+"="+drummond_combined[0][1]+"\\%}\\]";
|
||||
mathjax_for_prevalence_calcs+="\\[\\mathrm{Percent \\space of \\space untreated \\space with \\space YSS=\\frac{"
|
||||
+(untreated_men*10)+"\\space men *"
|
||||
+(your_m_utx_yss/100)+"\\space prev. +\\space "
|
||||
+((100-untreated_men)*10)+"\\space women *"
|
||||
+(your_f_utx_yss/100)+"\\space prev.}"
|
||||
+"{1000\\space total\\space untreated\\space patients}"
|
||||
+"="+drummond_combined[1][1]+"\\%}\\]";
|
||||
|
||||
$("#dynamic_mathjax").html(out_text);
|
||||
|
||||
MathJax.Hub.Queue(["Typeset",MathJax.Hub,"dynamic_mathjax"]);
|
||||
]]>
|
||||
</function>
|
||||
...
|
||||
|
||||
|
||||
Plot tag
|
||||
........
|
||||
|
||||
``Plot`` tag inside ``configuration`` tag defines settings for plot output.
|
||||
|
||||
Required parameters::
|
||||
|
||||
xrange: 2 functions that must return value. Value is constant (3.1415)
|
||||
or depend on parameter from parameters section:
|
||||
<xrange>
|
||||
<min>return 0;</min>
|
||||
<max>return 30;</max>
|
||||
</xrange>
|
||||
or
|
||||
<xrange>
|
||||
<min>return -a;</min>
|
||||
<max>return a;</max>
|
||||
</xrange>
|
||||
|
||||
All functions will be calculated over domain between xrange:min
|
||||
and xrange:max. Xrange depending on parameter is extremely
|
||||
useful when domain(s) of your function(s) depends on parameter
|
||||
(like circle, when parameter is radius and you want to allow
|
||||
to change it).
|
||||
|
||||
Optional parameters::
|
||||
|
||||
num_points: Number of data points to generated for the plot. If
|
||||
this is not set, the number of points will be
|
||||
calculated as width / 5.
|
||||
|
||||
bar_width: If functions are present which are to be plotted as bars,
|
||||
then this parameter specifies the width of the bars. A
|
||||
numeric value for this parameter is expected.
|
||||
|
||||
bar_align: If functions are present which are to be plotted as bars,
|
||||
then this parameter specifies how to align the bars relative
|
||||
to the tick. Available values are "left" and "center".
|
||||
|
||||
xticks,
|
||||
yticks: 3 floating point numbers separated by commas. This
|
||||
specifies how many ticks are created, what number they
|
||||
start at, and what number they end at. This is different
|
||||
from the 'xrange' setting in that it has nothing to do
|
||||
with the data points - it control what area of the
|
||||
Cartesian space you will see. The first number is the
|
||||
first tick's value, the second number is the step
|
||||
between each tick, the third number is the value of the
|
||||
last tick. If these configurations are not specified,
|
||||
Flot will chose them for you based on the data points
|
||||
set that he is currently plotting. Usually, this results
|
||||
in a nice graph, however, sometimes you need to fine
|
||||
grain the controls. For example, when you want to show
|
||||
a fixed area of the Cartesian space, even when the data
|
||||
set changes. On it's own, Flot will recalculate the
|
||||
ticks, which will result in a different graph each time.
|
||||
By specifying the xticks, yticks configurations, only
|
||||
the plotted data will change - the axes (ticks) will
|
||||
remain as you have defined them.
|
||||
|
||||
xticks_names, yticks_names:
|
||||
A JSON string which represents a mapping of xticks, yticks
|
||||
values to some defined strings. If specified, the graph will
|
||||
not have any xticks, yticks except those for which a string
|
||||
value has been defined in the JSON string. Note that the
|
||||
matching will be string-based and not numeric. I.e. if a tick
|
||||
value was "3.70" before, then inside the JSON there should be
|
||||
a mapping like {..., "3.70": "Some string", ...}. Example:
|
||||
|
||||
<xticks_names>
|
||||
<![CDATA[
|
||||
{
|
||||
"1": "Treated", "2": "Not Treated",
|
||||
"4": "Treated", "5": "Not Treated",
|
||||
"7": "Treated", "8": "Not Treated"
|
||||
}
|
||||
]]>
|
||||
</xticks_names>
|
||||
|
||||
<yticks_names>
|
||||
<![CDATA[
|
||||
{"0": "0%", "10": "10%", "20": "20%", "30": "30%", "40": "40%", "50": "50%"}
|
||||
]]>
|
||||
</yticks_names>
|
||||
|
||||
xunits,
|
||||
yunits: Units values to be set on axes. Use MathJax. Example:
|
||||
<xunits>\(cm\)</xunits>
|
||||
<yunits>\(m\)</yunits>
|
||||
|
||||
moving_label:
|
||||
A way to specify a label that should be positioned dynamically,
|
||||
based on the values of some parameters, or some other factors.
|
||||
It is similar to a <function>, but it is only valid for a plot
|
||||
because it is drawn relative to the plot coordinate system.
|
||||
|
||||
Multiple "moving_label" configurations can be provided, each one
|
||||
with a unique text and a unique set of functions that determine
|
||||
it's dynamic positioning.
|
||||
|
||||
Each "moving_label" can have a "color" attribute (CSS color notation),
|
||||
and a "weight" attribute. "weight" can be one of "normal" or "bold",
|
||||
and determines the styling of moving label's text.
|
||||
|
||||
Each "moving_label" function should return an object with a 'x'
|
||||
and 'y properties. Within those functions, all of the parameter
|
||||
names along with their value are available.
|
||||
|
||||
Example (note that "return" statement is missing; it will be automatically
|
||||
inserted by GST):
|
||||
|
||||
<moving_label text="Co" weight="bold" color="red>
|
||||
<![CDATA[ {'x': -50, 'y': c0};]]>
|
||||
</moving_label>
|
||||
|
||||
asymptote:
|
||||
Add a vertical or horizontal asymptote to the graph which will
|
||||
be dynamically repositioned based on the specified function.
|
||||
|
||||
It is similar to the function in that it provides a JavaScript body function
|
||||
string. This function will be used to calculate the position of the asymptote
|
||||
relative to the axis specified by the "type" parameter.
|
||||
|
||||
Required parameters:
|
||||
type:
|
||||
Which axis should the asymptote be plotted against. Available values
|
||||
are "x" and "y".
|
||||
|
||||
Optional parameters:
|
||||
color:
|
||||
The color of the line. A valid CSS color string is expected.
|
||||
|
||||
|
||||
Example
|
||||
=======
|
||||
|
||||
|
||||
Plotting, sliders and inputs
|
||||
----------------------------
|
||||
|
||||
.. literalinclude:: gst_example_with_documentation.xml
|
||||
|
||||
|
||||
Update of html elements, no plotting
|
||||
------------------------------------
|
||||
|
||||
.. literalinclude:: gst_example_html_element_output.xml
|
||||
|
||||
|
||||
Circle with dynamic radius
|
||||
--------------------------
|
||||
|
||||
.. literalinclude:: gst_example_dynamic_range.xml
|
||||
|
||||
|
||||
Example of a bar graph
|
||||
----------------------
|
||||
|
||||
.. literalinclude:: gst_example_bars.xml
|
||||
|
||||
|
||||
Example of moving labels of graph
|
||||
---------------------------------
|
||||
|
||||
.. literalinclude:: gst_example_dynamic_labels.xml
|
||||
@@ -0,0 +1,64 @@
|
||||
<vertical>
|
||||
<graphical_slider_tool>
|
||||
<render>
|
||||
<h2>Graphic slider tool: Bar graph example.</h2>
|
||||
|
||||
<p>We can request the API to plot us a bar graph.</p>
|
||||
<div style="clear:both">
|
||||
<p style="width:60px;float:left;">a</p>
|
||||
<slider var='a' style="width:400px;float:left;"/>
|
||||
<textbox var='a' style="width:50px;float:left;margin-left:15px;"/>
|
||||
<br /><br /><br />
|
||||
<p style="width:60px;float:left;">b</p>
|
||||
<slider var='b' style="width:400px;float:left;"/>
|
||||
<textbox var='b' style="width:50px;float:left;margin-left:15px;"/>
|
||||
</div>
|
||||
<plot style="clear:left;"/>
|
||||
</render>
|
||||
<configuration>
|
||||
<parameters>
|
||||
<param var="a" min="-100" max="100" step="5" initial="25" />
|
||||
<param var="b" min="-100" max="100" step="5" initial="50" />
|
||||
</parameters>
|
||||
<functions>
|
||||
<function bar="true" color="blue" label="Men">
|
||||
<![CDATA[if (((x>0.9) && (x<1.1)) || ((x>4.9) && (x<5.1))) { return Math.sin(a * 0.01 * Math.PI + 2.952 * x); }
|
||||
else {return undefined;}]]>
|
||||
</function>
|
||||
<function bar="true" color="red" label="Women">
|
||||
<![CDATA[if (((x>1.9) && (x<2.1)) || ((x>3.9) && (x<4.1))) { return Math.cos(b * 0.01 * Math.PI + 3.432 * x); }
|
||||
else {return undefined;}]]>
|
||||
</function>
|
||||
<function bar="true" color="green" label="Other 1">
|
||||
<![CDATA[if (((x>1.9) && (x<2.1)) || ((x>3.9) && (x<4.1))) { return Math.cos((b - 10 * a) * 0.01 * Math.PI + 3.432 * x); }
|
||||
else {return undefined;}]]>
|
||||
</function>
|
||||
<function bar="true" color="yellow" label="Other 2">
|
||||
<![CDATA[if (((x>1.9) && (x<2.1)) || ((x>3.9) && (x<4.1))) { return Math.cos((b + 7 * a) * 0.01 * Math.PI + 3.432 * x); }
|
||||
else {return undefined;}]]>
|
||||
</function>
|
||||
</functions>
|
||||
<plot>
|
||||
<xrange><min>1</min><max>5</max></xrange>
|
||||
<num_points>5</num_points>
|
||||
<xticks>0, 0.5, 6</xticks>
|
||||
<yticks>-1.5, 0.1, 1.5</yticks>
|
||||
<xticks_names>
|
||||
<![CDATA[
|
||||
{
|
||||
"1.5": "Single", "4.5": "Married"
|
||||
}
|
||||
]]>
|
||||
</xticks_names>
|
||||
<yticks_names>
|
||||
<![CDATA[
|
||||
{
|
||||
"-1.0": "-100%", "-0.5": "-50%", "0.0": "0%", "0.5": "50%", "1.0": "100%"
|
||||
}
|
||||
]]>
|
||||
</yticks_names>
|
||||
<bar_width>0.4</bar_width>
|
||||
</plot>
|
||||
</configuration>
|
||||
</graphical_slider_tool>
|
||||
</vertical>
|
||||
@@ -0,0 +1,40 @@
|
||||
<vertical>
|
||||
<graphical_slider_tool>
|
||||
<render>
|
||||
<h1>Graphic slider tool: Dynamic labels.</h1>
|
||||
<p>There are two kinds of dynamic lables.
|
||||
1) Dynamic changing values in graph legends.
|
||||
2) Dynamic labels, which coordinates depend on parameters </p>
|
||||
<p>a: <slider var="a"/></p>
|
||||
<br/>
|
||||
<p>b: <slider var="b"/></p>
|
||||
<br/><br/>
|
||||
<plot style="width:400px; height:400px;"/>
|
||||
</render>
|
||||
|
||||
<configuration>
|
||||
<parameters>
|
||||
<param var="a" min="-10" max="10" step="1" initial="0" />
|
||||
<param var="b" min="0" max="10" step="0.5" initial="5" />
|
||||
</parameters>
|
||||
<functions>
|
||||
<function label="Value of a is: dyn_val_1">a * x + b</function>
|
||||
<!-- dynamic values in legend -->
|
||||
<function output="plot_label" el_id="dyn_val_1">a</function>
|
||||
</functions>
|
||||
<plot>
|
||||
<xrange><min>0</min><max>30</max></xrange>
|
||||
<num_points>10</num_points>
|
||||
<xticks>0, 6, 30</xticks>
|
||||
<yticks>-9, 1, 9</yticks>
|
||||
<!-- custom labels with coordinates as any function of parameter -->
|
||||
<moving_label text="Dynam_lbl 1" weight="bold">
|
||||
<![CDATA[ {'x': 10, 'y': a};]]>
|
||||
</moving_label>
|
||||
<moving_label text="Dynam lbl 2" weight="bold">
|
||||
<![CDATA[ {'x': -6, 'y': b};]]>
|
||||
</moving_label>
|
||||
</plot>
|
||||
</configuration>
|
||||
</graphical_slider_tool>
|
||||
</vertical>
|
||||
@@ -0,0 +1,37 @@
|
||||
<vertical>
|
||||
<graphical_slider_tool>
|
||||
<render>
|
||||
<h2>Graphic slider tool: Dynamic range and implicit functions.</h2>
|
||||
|
||||
<p>You can make x range (not ticks of x axis) of functions to depend on
|
||||
parameter value. This can be useful when function domain depends
|
||||
on parameter.</p>
|
||||
<p>Also implicit functons like circle can be plotted as 2 separate
|
||||
functions of same color.</p>
|
||||
<div style="height:50px;">
|
||||
<slider var='a' style="width:400px;float:left;"/>
|
||||
<textbox var='a' style="float:left;width:60px;margin-left:15px;"/>
|
||||
</div>
|
||||
<plot style="margin-top:15px;margin-bottom:15px;"/>
|
||||
</render>
|
||||
<configuration>
|
||||
<parameters>
|
||||
<param var="a" min="5" max="25" step="0.5" initial="12.5" />
|
||||
</parameters>
|
||||
<functions>
|
||||
<function color="red">Math.sqrt(a * a - x * x)</function>
|
||||
<function color="red">-Math.sqrt(a * a - x * x)</function>
|
||||
</functions>
|
||||
<plot>
|
||||
<xrange>
|
||||
<!-- dynamic range -->
|
||||
<min>-a</min>
|
||||
<max>a</max>
|
||||
</xrange>
|
||||
<num_points>1000</num_points>
|
||||
<xticks>-30, 6, 30</xticks>
|
||||
<yticks>-30, 6, 30</yticks>
|
||||
</plot>
|
||||
</configuration>
|
||||
</graphical_slider_tool>
|
||||
</vertical>
|
||||
@@ -0,0 +1,40 @@
|
||||
<vertical>
|
||||
<graphical_slider_tool>
|
||||
<render>
|
||||
<h2>Graphic slider tool: Output to DOM element.</h2>
|
||||
|
||||
<p>a + b = <span id="answer_span_1"></span></p>
|
||||
|
||||
<div style="clear:both">
|
||||
<p style="float:left;margin-right:10px;">a</p>
|
||||
<slider var='a' style="width:400px;float:left;"/>
|
||||
<textbox var='a' style="width:50px;float:left;margin-left:15px;"/>
|
||||
</div>
|
||||
|
||||
<div style="clear:both">
|
||||
<p style="float:left;margin-right:10px;">b</p>
|
||||
<slider var='b' style="width:400px;float:left;"/>
|
||||
<textbox var='b' style="width:50px;float:left;margin-left:15px;"/>
|
||||
</div>
|
||||
<br/><br/><br/>
|
||||
<plot/>
|
||||
</render>
|
||||
<configuration>
|
||||
<parameters>
|
||||
<param var="a" min="-10.0" max="10.0" step="0.1" initial="0" />
|
||||
<param var="b" min="-10.0" max="10.0" step="0.1" initial="0" />
|
||||
</parameters>
|
||||
|
||||
<functions>
|
||||
<function output="element" el_id="answer_span_1">
|
||||
function add(a, b, precision) {
|
||||
var x = Math.pow(10, precision || 2);
|
||||
return (Math.round(a * x) + Math.round(b * x)) / x;
|
||||
}
|
||||
|
||||
return add(a, b, 5);
|
||||
</function>
|
||||
</functions>
|
||||
</configuration>
|
||||
</graphical_slider_tool>
|
||||
</vertical>
|
||||
@@ -0,0 +1,91 @@
|
||||
<vertical>
|
||||
<graphical_slider_tool>
|
||||
<render>
|
||||
|
||||
<h2>Graphic slider tool: full example.</h2>
|
||||
<p>
|
||||
A simple equation
|
||||
\(
|
||||
y_1 = 10 \times b \times \frac{sin(a \times x) \times sin(b \times x)}{cos(b \times x) + 10}
|
||||
\)
|
||||
can be plotted.
|
||||
</p>
|
||||
|
||||
<!-- make text and input or slider at the same line -->
|
||||
<div>
|
||||
<p style="float:left;"> Currently \(a\) is</p>
|
||||
<!-- readonly input for a -->
|
||||
<span id="a_readonly" style="width:50px; float:left; margin-left:10px;"/>
|
||||
</div>
|
||||
<!-- clear:left will make next text to begin from new line -->
|
||||
<p style="clear:left"> This one
|
||||
\(
|
||||
y_2 = sin(a \times x)
|
||||
\)
|
||||
will be overlayed on top.
|
||||
</p>
|
||||
<div>
|
||||
<p style="float:left;">Currently \(b\) is </p>
|
||||
<textbox var="b" style="width:50px; float:left; margin-left:10px;"/>
|
||||
</div>
|
||||
<div style="clear:left;">
|
||||
<p style="float:left;">To change \(a\) use:</p>
|
||||
<slider var="a" style="width:400px;float:left;margin-left:10px;"/>
|
||||
</div>
|
||||
<div style="clear:left;">
|
||||
<p style="float:left;">To change \(b\) use:</p>
|
||||
<slider var="b" style="width:400px;float:left;margin-left:10px;"/>
|
||||
</div>
|
||||
<plot style='clear:left;width:600px;padding-top:15px;padding-bottom:20px;'/>
|
||||
<div style="clear:left;height:50px;">
|
||||
<p style="float:left;">Second input for b:</p>
|
||||
<!-- editable input for b -->
|
||||
<textbox var="b" style="color:red;width:60px;float:left;margin-left:10px;"/>
|
||||
</div >
|
||||
</render>
|
||||
|
||||
<configuration>
|
||||
|
||||
<parameters>
|
||||
<param var="a" min="90" max="120" step="10" initial="100" />
|
||||
<param var="b" min="120" max="200" step="2.3" initial="120" />
|
||||
</parameters>
|
||||
|
||||
<functions>
|
||||
|
||||
<function color="#0000FF" line="false" dot="true" label="\(y_1\)">
|
||||
return 10.0 * b * Math.sin(a * x) * Math.sin(b * x) / (Math.cos(b * x) + 10);
|
||||
</function>
|
||||
<function color="red" line="true" dot="false" label="\(y_2\)">
|
||||
<!-- works w/o return, if function is single line -->
|
||||
Math.sin(a * x);
|
||||
</function>
|
||||
<function color="#FFFF00" line="false" dot="false" label="unknown">
|
||||
function helperFunc(c1) {
|
||||
return c1 * c1 - a;
|
||||
}
|
||||
|
||||
return helperFunc(x + 10 * a * b) + Math.sin(a - x);
|
||||
</function>
|
||||
<function output="element" el_id="a_readonly">a</function>
|
||||
</functions>
|
||||
|
||||
<plot>
|
||||
|
||||
<xrange>
|
||||
<min>return 0;</min>
|
||||
<!-- works w/o return -->
|
||||
<max>30</max>
|
||||
</xrange>
|
||||
|
||||
<num_points>120</num_points>
|
||||
|
||||
<xticks>0, 3, 30</xticks>
|
||||
<yticks>-1.5, 1.5, 13.5</yticks>
|
||||
|
||||
<xunits>\(cm\)</xunits>
|
||||
<yunits>\(m\)</yunits>
|
||||
</plot>
|
||||
</configuration>
|
||||
</graphical_slider_tool>
|
||||
</vertical>
|
||||
@@ -0,0 +1,67 @@
|
||||
**********************************************
|
||||
Xml format of poll module [xmodule]
|
||||
**********************************************
|
||||
|
||||
.. module:: poll_module
|
||||
|
||||
Format description
|
||||
==================
|
||||
|
||||
The main tag of Poll module input is:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<poll_question> ... </poll_question>
|
||||
|
||||
``poll_question`` can include any number of the following tags:
|
||||
any xml and ``answer`` tag. All inner xml, except for ``answer`` tags, we call "question".
|
||||
|
||||
poll_question tag
|
||||
-----------------
|
||||
|
||||
Xmodule for creating poll functionality - voting system. The following attributes can
|
||||
be specified for this tag::
|
||||
|
||||
name - Name of xmodule.
|
||||
[display_name| AUTOGENERATE] - Display name of xmodule. When this attribute is not defined - display name autogenerate with some hash.
|
||||
[reset | False] - Can reset/revote many time (value = True/False)
|
||||
|
||||
|
||||
answer tag
|
||||
----------
|
||||
|
||||
Define one of the possible answer for poll module. The following attributes can
|
||||
be specified for this tag::
|
||||
|
||||
id - unique identifier (using to identify the different answers)
|
||||
|
||||
Inner text - Display text for answer choice.
|
||||
|
||||
Example
|
||||
=======
|
||||
|
||||
Examples of poll
|
||||
----------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<poll_question name="second_question" display_name="Second question">
|
||||
<h3>Age</h3>
|
||||
<p>How old are you?</p>
|
||||
<answer id="less18">< 18</answer>
|
||||
<answer id="10_25">from 10 to 25</answer>
|
||||
<answer id="more25">> 25</answer>
|
||||
</poll_question>
|
||||
|
||||
Examples of poll with unable reset functionality
|
||||
------------------------------------------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<poll_question name="first_question_with_reset" display_name="First question with reset"
|
||||
reset="True">
|
||||
<h3>Your gender</h3>
|
||||
<p>You are man or woman?</p>
|
||||
<answer id="man">Man</answer>
|
||||
<answer id="woman">Woman</answer>
|
||||
</poll_question>
|
||||
40
docs/data/source/course_data_formats/symbolic_response.rst
Normal file
40
docs/data/source/course_data_formats/symbolic_response.rst
Normal file
@@ -0,0 +1,40 @@
|
||||
#################
|
||||
Symbolic Response
|
||||
#################
|
||||
|
||||
This document plans to document features that the current symbolic response
|
||||
supports. In general it allows the input and validation of math expressions,
|
||||
up to commutativity and some identities.
|
||||
|
||||
|
||||
********
|
||||
Features
|
||||
********
|
||||
|
||||
This is a partial list of features, to be revised as we go along:
|
||||
* sub and superscripts: an expression following the ``^`` character
|
||||
indicates exponentiation. To use superscripts in variables, the syntax
|
||||
is ``b_x__d`` for the variable ``b`` with subscript ``x`` and super
|
||||
``d``.
|
||||
|
||||
An example of a problem::
|
||||
|
||||
<symbolicresponse expect="a_b^c + b_x__d" size="30">
|
||||
<textline math="1"
|
||||
preprocessorClassName="SymbolicMathjaxPreprocessor"
|
||||
preprocessorSrc="/static/js/capa/symbolic_mathjax_preprocessor.js"/>
|
||||
</symbolicresponse>
|
||||
|
||||
It's a bit of a pain to enter that.
|
||||
|
||||
* The script-style math variant. What would be outputted in latex if you
|
||||
entered ``\mathcal{N}``. This is used in some variables.
|
||||
|
||||
An example::
|
||||
|
||||
<symbolicresponse expect="scriptN_B + x" size="30">
|
||||
<textline math="1"/>
|
||||
</symbolicresponse>
|
||||
|
||||
There is no fancy preprocessing needed, but if you had superscripts or
|
||||
something, you would need to include that part.
|
||||
BIN
docs/data/source/course_data_formats/word_cloud/word_cloud.png
Normal file
BIN
docs/data/source/course_data_formats/word_cloud/word_cloud.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
@@ -0,0 +1,59 @@
|
||||
**********************************************
|
||||
Xml format of "Word Cloud" module [xmodule]
|
||||
**********************************************
|
||||
|
||||
.. module:: word_cloud
|
||||
|
||||
Format description
|
||||
==================
|
||||
|
||||
The main tag of Word Cloud module input is:
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<word_cloud />
|
||||
|
||||
The following attributes can be specified for this tag::
|
||||
|
||||
[display_name| AUTOGENERATE] – Display name of xmodule. When this attribute is not defined - display name autogenerate with some hash.
|
||||
[num_inputs| 5] – Number of inputs.
|
||||
[num_top_words| 250] – Number of max words, which will be displayed.
|
||||
[display_student_percents| True] – Display usage percents for each word on the same line together with words.
|
||||
|
||||
.. note::
|
||||
|
||||
Percent is shown always when mouse over the word in cloud.
|
||||
|
||||
.. note::
|
||||
|
||||
Possible answer for boolean type attributes:
|
||||
True – "True", "true", "T", "t", "1"
|
||||
False – "False", "false", "F", "f", "0"
|
||||
|
||||
.. note::
|
||||
|
||||
If you want to use the same word cloud (the same storage of words), you must use the same display_name value.
|
||||
|
||||
|
||||
Code Example
|
||||
============
|
||||
|
||||
Examples of word_cloud without all attributes (all attributes get by default)
|
||||
-----------------------------------------------------------------------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<word_cloud />
|
||||
|
||||
Examples of poll with all attributes
|
||||
------------------------------------
|
||||
|
||||
.. code-block:: xml
|
||||
|
||||
<word_cloud display_name="cloud" num_inputs="10" num_top_words="100" />
|
||||
|
||||
Screenshots
|
||||
===========
|
||||
|
||||
.. image:: word_cloud.png
|
||||
:width: 50%
|
||||
Reference in New Issue
Block a user