Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
21 changes: 19 additions & 2 deletions common/lib/xmodule/xmodule/textannotation_module.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,8 +47,24 @@ class AnnotatableFields(object):
scope=Scope.settings,
default='None',
)
annotation_storage_url = String(help=_("Location of Annotation backend"), scope=Scope.settings, default="http://your_annotation_storage.com", display_name=_("Url for Annotation Storage"))
annotation_token_secret = String(help=_("Secret string for annotation storage"), scope=Scope.settings, default="xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx", display_name=_("Secret Token String for Annotation"))
annotation_storage_url = String(
help=_("Location of Annotation backend"),
scope=Scope.settings,
default="http://your_annotation_storage.com",
display_name=_("Url for Annotation Storage"),
)
annotation_token_secret = String(
help=_("Secret string for annotation storage"),
scope=Scope.settings,
default="xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
display_name=_("Secret Token String for Annotation"),
)
diacritics = String(
display_name=_("Diacritic Marks"),
help=_("Add diacritic marks to be added to a text using the comma-separated form, i.e. markname;urltomark;baseline,markname2;urltomark2;baseline2"),
scope=Scope.settings,
default='',
)


class TextAnnotationModule(AnnotatableFields, XModule):
Expand Down Expand Up @@ -84,6 +100,7 @@ def student_view(self, context):
'content_html': self.content,
'annotation_storage': self.annotation_storage_url,
'token': retrieve_token(self.user_email, self.annotation_token_secret),
'diacritic_marks': self.diacritics,
}
fragment = Fragment(self.system.render_template('textannotation.html', context))
fragment.add_javascript_url("/static/js/vendor/tinymce/js/tinymce/tinymce.full.min.js")
Expand Down
27 changes: 23 additions & 4 deletions common/lib/xmodule/xmodule/videoannotation_module.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,10 +34,29 @@ class AnnotatableFields(object):
scope=Scope.settings,
default=_('Video Annotation'),
)
sourceurl = String(help=_("The external source URL for the video."), display_name=_("Source URL"), scope=Scope.settings, default="http://video-js.zencoder.com/oceans-clip.mp4")
poster_url = String(help=_("Poster Image URL"), display_name=_("Poster URL"), scope=Scope.settings, default="")
annotation_storage_url = String(help=_("Location of Annotation backend"), scope=Scope.settings, default="http://your_annotation_storage.com", display_name=_("Url for Annotation Storage"))
annotation_token_secret = String(help=_("Secret string for annotation storage"), scope=Scope.settings, default="xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx", display_name=_("Secret Token String for Annotation"))
sourceurl = String(
help=_("The external source URL for the video."),
display_name=_("Source URL"),
scope=Scope.settings, default="http://video-js.zencoder.com/oceans-clip.mp4"
)
poster_url = String(
help=_("Poster Image URL"),
display_name=_("Poster URL"),
scope=Scope.settings,
default=""
)
annotation_storage_url = String(
help=_("Location of Annotation backend"),
scope=Scope.settings,
default="http://your_annotation_storage.com",
display_name=_("Url for Annotation Storage"),
)
annotation_token_secret = String(
help=_("Secret string for annotation storage"),
scope=Scope.settings,
default="xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx",
display_name=_("Secret Token String for Annotation")
)

class VideoAnnotationModule(AnnotatableFields, XModule):
'''Video Annotation Module'''
Expand Down
8 changes: 8 additions & 0 deletions common/static/css/vendor/ova/diacritic-annotator.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
.mark{
width: 10px;
height: 10px;
position: absolute;
background-size: contain;
background-repeat: no-repeat;
background-position: 50% 0%;
}
4 changes: 4 additions & 0 deletions common/static/css/vendor/ova/edx-annotator.css
Original file line number Diff line number Diff line change
@@ -1,4 +1,8 @@
/*This is written to fix some design problems with edX*/
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi, would you be more specific about what UI problems this resolves? I'm hoping others who come across this comment can see exactly where and why this is needed (and that perhaps the root problem can be addressed down the road).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's specific only to the annotator tool we've created as an XModule. There is an editor that pops up to take annotation and the rules here solve design problems with mixing annotator.css within the standard edx css.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The rules are all specific to the annotator tool (all contain the class .annotator or .annotator-wrapper and its variants) so it should not affect the UI outside of this context.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, can you make the comment you added the code summarize the specific issues you're resolving due to edX's LMS base CSS (e.g. "These rules are needed to make sure the annotation UI 'X', 'Y', and 'Z' elements within the edX LMS context")? I'd like future developers/designers to know what exactly you had to overcompensate for here in the UI.

.annotatable-wrapper .annotatable-header .annotatable-title{
padding-top: 20px !important;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we really need this !important? Can we re-write it without it?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

++ We try to not use !important anywhere in our css.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So how do I get around the fact that courses.css overwrites the styles in edx-annotator.css?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there also used !important ?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wait I'm so confused about what your'e telling me to do...
As an example:

  1. My CSS rule is for .annotator-editor a, .annotator-filter .annotator-filter-property label. If I take the !important out of this rule and all rules on this CSS file then the rules from courses.css overwrite it.
  2. The classes in courses.css all have rules like a:link, a:visited which are LESS specific than mine.
  3. Because they are more broad they overwrite my more specific rules
  4. According to that article it tells me only to use !important when I'm being specific in my attributes and once again, my rules are more specific than those on courses.css.
  5. I didn't write courses.css and those seem to involve higher and broader standards should I not provide my specific changes. Am I just being dumb and putting my css in the wrong place where it will only get loaded before and not after?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi. Perhaps I can be of some help here.

I agree with others that we don't want to use !important in our CSS rules if at all possible (it creates a styling shouting match that is near impossible to maintain over time). @polesye's link to CSS specificity is right on. Though courses.css is loaded/parsed after your CSS file, there are ways to have your rule be more specific and thus respected when there are two conflicted values for the same property of the same element.

The rules in the line we've commented on are different than the ones noted in your comment above (.annotatable-wrapper .annotatable-header .annotatable-title vs. .annotator-editor a, .annotator-filter .annotator-filter-property label). Assuming we're dealing with the selectors from your actual code, without knowing exactly what is overriding your padding on .annotatable-wrapper .annotatable-header .annotatable-title, my steps to resolve this would be:

  1. Find out what base LMS style rule (from courses.css) is trumping yours. This involves peeking at the web inspector and finding the exact selector rule and the property involved.
  2. Re-craft your CSS selector to be more specific than the rule you found in step 1. This could involve adding additional pseudo states such as :visisited, :link, :active if you're working with links or could be as easy as referring to the parent element's class attribute or a specific HTML element.
  3. Remove the !important and comment in detail on why the style rule is needed. These will help with maintenance and others who want to build on what you have or pay down some CSS debt and remove the hurdle you jumped over.

if you still need help, find the specific style rule that's overriding yours and share it here. We can craft a selector that can get the job done from there.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@polesye and @talbs - Quick clarification:
So you're not against all of the !important in this file? Just the one specific line you've commented on? same with richText-annotator.css?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My ideal outcome is that we do not introduce any !important statements into the CSS. You should be able to be more specific with selectors in all cases to overcome the LMS CSS.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@talbs Is there any way that we can schedule a video/audio call to discuss the matter?

}

.annotator-wrapper .annotator-adder button {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is this needed? If you're visually hiding the element in the UI, you'll need to consider how screen readers and assistive tech will handle it (in this case I believe they'll still announce and present the button). Toggling an attribute of aria-hidden="true/false" (true for when something is visually hidden and false for when it should be shown/available) in your HTML (through JavaScript) will sync up semantics with your visual UI state.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We're not hiding the button. The background image gets covered up by a gradient in the background (gray colored) and setting the opacity to 0 is the only way to cover up the color and leave show the actual background image.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the clarification. Carry on :)

opacity:0;
}
Expand Down
14 changes: 9 additions & 5 deletions common/static/css/vendor/ova/richText-annotator.css
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,14 @@
font-style: italic;
}

.mce-floatpanel {
z-index: 700000000!important;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I also agree with @polesye's concern over the use of the large and arbitrary z-index here. While I know you need to set a value that will bring these elements to the top of the UI depth-wise, I'd recommend using a smaller number and also adhere to a scale that others can understand.

We happen to have Sass placeholders elsewhere that adhere to a depth scale (see https://github.com/edx/edx-platform/blob/master/lms/static/sass/base/_mixins.scss#L93). Since you're writing CSS and not Sass, I'd recommend just borrowing the appropriate z-index value (e.g. z-index: 1000;) from that scale. Ideally, we want to not use the largest value as other UI elements may need to take front stage (such as modal windows, prompts, warnings, etc.).

}

.annotator-wrapper .mce-container {
z-index: 3000000000!important; /*To fix full-screen problems*/
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Again, try to avoid !important in these two rules, please.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we really need so big z-index?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would second the strong recommendation to avoid using !important here as well. Please see my previous comments on how to figure out being more specific instead.

}

.mce-container-body {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are all of these min-width properties needed? What happens to the UI if it just takes up the width/dimensions of its parent? If possible, I'd rather rely on that context rather than have several widths that are set manually here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This code sets up a "popup" editor and viewer to annotate different sections of text. This code is also extending part of a vendor file. The vendor sets min-widths that are way too thin and we wanted to extend them.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Again, I feel like a call would be best for me to describe the situation. I don't feel like you've quite grasped the context into which the code is added. I understand that there are rules and preferences in terms of how it should all be formatted and whatnot, but there are limitations to things I can do when something like a vendor file is involved.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks again for the additional info. Knowing the context, this seems fine and we can disregard my initial concern.

min-width: 400px;
}
Expand All @@ -25,10 +33,6 @@
min-width: 400px;
}

.mce-floatpanel {
z-index: 700000000!important;
}

div.mce-tinymce.mce-container.mce-panel {
min-width:400px;
}
Expand All @@ -42,4 +46,4 @@ div.mce-tinymce.mce-container.mce-panel {
.mce-ico.mce-i-rubric{
background-image: url('');
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where is this image assets and how is it being loaded? Please leave a comment at the least around what this is referencing (since the source file syntax/name doesn't clearly explain that).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The image asset is right there?

As the url states, we've encoded an "image" in "png" format using "base64" and are including the "data:" followed by the encoded data. I'm not sure I understand the question?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the information. This format/method of referencing a background-image value is a bit less common and more obfuscated (i.e. tougher for future editors to understand where the asset is located and what specifically it is).

background-repeat: no-repeat;
}
}
2 changes: 1 addition & 1 deletion common/static/css/vendor/ova/tags-annotator.css
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ li.token-input-input-token {

div.token-input-dropdown {
position: absolute;
width: 120px;
width: 400px;
background-color: #fff;
overflow: hidden;
border-left: 1px solid #ccc;
Expand Down
270 changes: 270 additions & 0 deletions common/static/js/vendor/ova/diacritic-annotator.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,270 @@
/*
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. Please fix alignment in this file.
  2. Max line width is 80 characters.
  3. Use 1 keyword var per function.
  4. Use JSHint.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. Add docstrings. Use jsDoc.

Diacritic Annotator Plugin v1.0 (https://github.com/lduarte1991/diacritic-annotator)
Copyright (C) 2014 Luis F Duarte
License: https://github.com/lduarte1991/diacritic-annotator/blob/master/LICENSE.rst

This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/

var _ref;
var __bind = function(fn, me){
return function(){
return fn.apply(me, arguments);
};
};
var __hasProp = {}.hasOwnProperty;
var __extends = function(child, parent) {
for (var key in parent) {
if (__hasProp.call(parent, key))
child[key] = parent[key];
}
function ctor() { this.constructor = child; }
ctor.prototype = parent.prototype;
child.prototype = new ctor();
child.__super__ = parent.prototype;
return child;
};

Annotator.Plugin.Diacritics = (function(_super) {
__extends(Diacritics, _super);

//Options will include diacritic name, picture used, baseline
Diacritics.prototype.options = null;
Diacritics.prototype.diacriticmarks = null;

/**
* Declares all the functions and variables that the plugin will need.
* @constructor
*/
function Diacritics(element,options) {
this.pluginSubmit = __bind(this.pluginSubmit, this);
this.updateDiacritics = __bind(this.updateDiacritics, this);
this.updateViewer = __bind(this.updateViewer, this);
this.getDiacritics = __bind(this.getDiacritics, this);
this.getPos = __bind(this.getPos, this);
this.putMarkAtLocation = __bind(this.putMarkAtLocation, this);
this.updateEditorForDiacritics =
__bind(this.updateEditorForDiacritics, this);

this.options = options;
this.diacriticmarks = this.getDiacritics();
_ref = Diacritics.__super__.constructor.apply(this, arguments);
return _ref;
}

//example variables to be used to receive input in the annotator view
Diacritics.prototype.field = null;
Diacritics.prototype.input = null;

/**
* Initalizes the Plug-in for diacritic marks. It adds in the field for the mark
* and sets up listeners from the Annotator.js file to make changes as needed
*/
Diacritics.prototype.pluginInit = function() {
//Check that annotator is working
if (!Annotator.supported()) {
return;
}
var di = this.diacriticmarks;

//-- Editor
var self = this;
if(di != 'undefined'){
$.each(di,function(item){
self.field = self.annotator.editor.addField({
//options (textarea,input,select,checkbox)
type: 'checkbox',
label: Annotator._t(item),
submit: self.pluginSubmit,
});
});

//-- Viewer
this.annotator.viewer.addField({
load: this.updateViewer,
});

this.annotator.subscribe('annotationsLoaded', this.updateDiacritics);
this.annotator.subscribe('annotationUploaded', this.updateDiacritics);
this.annotator.subscribe('annotationDeleted', this.updateDiacritics);
this.annotator.subscribe('annotationUpdated', this.updateDiacritics);
this.annotator.subscribe('annotationEditorShown', this.updateEditorForDiacritics, this.field);

$(window).resize(this.updateDiacritics.bind(this));
}

return this.input = $(this.field).find(':input');
};

/**
* Adds or removes tag from checked/unchecked boxes of diacritics available
* @param field {Object} - element which holds editor
* @param annotation {Object} - object that contains annotation information from database
*/
Diacritics.prototype.pluginSubmit = function(field, annotation) {
var checkedItems = $(this.field).find(':input');
var self = this;
$.each(checkedItems, function(item){
if(typeof annotation.tags != 'undefined'){
var index = $.inArray(checkedItems[item].placeholder, annotation.tags);
if(index != -1){
annotation.tags.splice(index, 1);
var annotatorWrapper = $('.annotator-wrapper').first();
var element = annotatorWrapper.find('div.' + annotation.id);

if(!element.length){
element = annotatorWrapper.find('div.undefined');
}

element.remove();
}

if(checkedItems[item].checked === true){
annotation.tags.unshift(checkedItems[item].placeholder);
self.putMarkAtLocation(annotation, checkedItems[item].placeholder);
}
} else {
if(checkedItems[item].checked === true){
annotation.tags = [checkedItems[item].placeholder];
self.putMarkAtLocation(annotation, checkedItems[item].placeholder);
}
}
});

};

/**
* Draws the mark above a specific annotation
* @param annotation {Object} - location where mark should go
* @param mark {string}- type of mark that should go above annotation
*/
Diacritics.prototype.putMarkAtLocation = function (annotation, mark){
var loc = this.getPos(annotation.highlights[0]);
var alignment = this.diacriticmarks[mark][1];
var imgurl = this.diacriticmarks[mark][0];

var top;
switch(alignment){
case 'top':
top = (loc.y-5);
break;
case 'bottom':
top = (loc.y + loc.height-5);
break;
default:
top = loc.y;
}
$('<div></div>').addClass('mark ' + annotation.id).css({
'top': top,
'left': loc.x + (0.5 * loc.width) - 5,
'background-image': 'url(' + imgurl +')',
}).appendTo('.annotator-wrapper');
};

/**
* Gets the Diacritics from the instantiation in studio
* @returns An object with the diacritics instantiated
*/
Diacritics.prototype.getDiacritics = function(){
var diacritics = {};
var diacriticsList;
if(typeof this.options.diacritics != 'undefined'){
diacriticsList = this.options.diacritics.split(",");
$.each(diacriticsList, function(key, item){
var temp = item.split(";");
if (temp.length > 2) {
diacritics[temp[0]] = [temp[1], temp[2]];
}
});
}
return diacritics;
};

/**
* Gets the position of a specific element given the wrapper
* @param el {Object} - element you are trying to get the position of
*/
Diacritics.prototype.getPos = function(el) {
var element = $(el),
elementOffset = element.offset(),
annotatorOffset = $('.annotator-wrapper').offset();

return {
x: elementOffset.left - annotatorOffset.left,
y: elementOffset.top - annotatorOffset.top,
width: element.width(),
height: element.height()
};
};

/**
* Redraws the marks above annotations by cycling through tags
*/
Diacritics.prototype.updateDiacritics = function(){
$('.mark').remove();
var annotations = this.annotator.plugins.Store.annotations;
var self = this;
$.each(annotations, function(key, annotation){
$.each(self.diacriticmarks, function(tag){
if($.inArray(tag, annotation.tags) != -1){
self.putMarkAtLocation(annotation, tag);
}
});
});
};

/**
* Removes unnecessary field that Annotator automatically adds to popup
* @param {Object} field - the html element that represents the popup
* @param {Object} annotation - the annotation element that holds metadata
*/
Diacritics.prototype.updateViewer = function(field, annotation){
$(field).remove();
};

/**
* Function for adding Diacritic choices to the annotator popup
* @param {Object} field - the html element that represents the popup
* @param {Object} annotation - the annotation element that holds metadata
*/
Diacritics.prototype.updateEditorForDiacritics =
function(field, annotation){

// if no tags are present, no need to go through this
if (typeof annotation.tags == 'undefined'){
return;
}

var inputItems = $(this.field).find(':input');
var dictOfItems = {};
var self = this;

// add each diacritic mark to a dictionary and default to off
$.each(inputItems, function(key,item){
item.checked = false;
dictOfItems[item.placeholder] = item;
});

// match tags to diacritics and check off the ones that are true
$.each(annotation.tags, function(key,tag){
if(self.diacriticmarks[tag]){
dictOfItems[tag].checked = true;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do we need dictOfItems? Where does it used?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In 262 it's created to have the name of the tag be the key. Then in 268 it is used as an easier way to set the checkbox on if that diacritic mark is already a tag.

}
});
};

return Diacritics;

})(Annotator.Plugin);
Loading