181 lines
6.4 KiB
Ruby
181 lines
6.4 KiB
Ruby
#
|
|
# Copyright (C) 2011 - present Instructure, Inc.
|
|
#
|
|
# This file is part of Canvas.
|
|
#
|
|
# Canvas is free software: you can redistribute it and/or modify it under
|
|
# the terms of the GNU Affero General Public License as published by the Free
|
|
# Software Foundation, version 3 of the License.
|
|
#
|
|
# Canvas 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 Affero General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License along
|
|
# with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
class LearningOutcomeResult < ActiveRecord::Base
|
|
belongs_to :user
|
|
belongs_to :learning_outcome
|
|
belongs_to :alignment, :class_name => 'ContentTag', :foreign_key => :content_tag_id
|
|
belongs_to :association_object, polymorphic:
|
|
[:rubric_association, :assignment,
|
|
{ quiz: 'Quizzes::Quiz', assessment: 'LiveAssessments::Assessment' }],
|
|
polymorphic_prefix: :association,
|
|
foreign_type: :association_type, foreign_key: :association_id
|
|
belongs_to :artifact, polymorphic:
|
|
[:rubric_assessment, :submission,
|
|
{ quiz_submission: 'Quizzes::QuizSubmission', live_assessments_submission: 'LiveAssessments::Submission' }],
|
|
polymorphic_prefix: true
|
|
belongs_to :associated_asset, polymorphic:
|
|
[:assessment_question, :assignment,
|
|
{ quiz: 'Quizzes::Quiz', assessment: 'LiveAssessments::Assessment' }],
|
|
polymorphic_prefix: true
|
|
belongs_to :context, polymorphic: [:course]
|
|
has_many :learning_outcome_question_results, dependent: :destroy
|
|
simply_versioned
|
|
|
|
before_save :infer_defaults
|
|
|
|
def calculate_percent!
|
|
scale_data = scale_params
|
|
if needs_scale?(scale_data) && self.score && self.possible
|
|
self.percent = calculate_by_scale(scale_data).round(4)
|
|
elsif self.score && self.possible
|
|
self.percent = percentage
|
|
end
|
|
self.percent = nil if self.percent && !self.percent.to_f.finite?
|
|
end
|
|
|
|
def percentage
|
|
if self.possible.to_f > 0
|
|
self.score.to_f / self.possible.to_f
|
|
elsif parent_has_mastery?
|
|
# the parent should always have a mastery score, if it doesn't
|
|
# it means something is broken with the outcome and it will need to
|
|
# be corrected. If percent is nil on an Outcome Result associated with
|
|
# a Quiz, it will cause a 500 error on the learning mastery gradebook in
|
|
# the get_aggregates method in RollupScoreAggregatorHelper
|
|
self.score.to_f / parent_outcome.mastery_points.to_f
|
|
end
|
|
end
|
|
|
|
def assignment
|
|
if self.association_object.is_a?(Assignment)
|
|
self.association_object
|
|
elsif self.artifact.is_a?(RubricAssessment)
|
|
self.artifact.rubric_association.association_object
|
|
else
|
|
nil
|
|
end
|
|
end
|
|
|
|
def save_to_version(attempt)
|
|
current_version = self.versions.current.try(:model)
|
|
if current_version.try(:attempt) && attempt < current_version.attempt
|
|
versions = self.versions.sort_by(&:created_at).reverse.select{|v| v.model.attempt == attempt}
|
|
if !versions.empty?
|
|
versions.each do |version|
|
|
version_data = YAML::load(version.yaml)
|
|
version_data["score"] = self.score
|
|
version_data["mastery"] = self.mastery
|
|
version_data["possible"] = self.possible
|
|
version_data["attempt"] = self.attempt
|
|
version_data["title"] = self.title
|
|
version.yaml = version_data.to_yaml
|
|
version.save
|
|
end
|
|
else
|
|
save
|
|
end
|
|
else
|
|
save
|
|
end
|
|
end
|
|
|
|
def submitted_or_assessed_at
|
|
submitted_at || assessed_at
|
|
end
|
|
|
|
scope :for_context_codes, lambda { |codes|
|
|
if codes == 'all'
|
|
all
|
|
else
|
|
where(:context_code => codes)
|
|
end
|
|
}
|
|
scope :for_user, lambda { |user| where(:user_id => user) }
|
|
scope :custom_ordering, lambda { |param|
|
|
orders = {
|
|
'recent' => "assessed_at DESC",
|
|
'highest' => "score DESC",
|
|
'oldest' => "score ASC",
|
|
'default' => "assessed_at DESC"
|
|
}
|
|
order_clause = orders[param] || orders['default']
|
|
order(order_clause)
|
|
}
|
|
scope :for_outcome_ids, lambda { |ids| where(:learning_outcome_id => ids) }
|
|
scope :for_association, lambda { |association| where(:association_type => association.class.to_s, :association_id => association.id) }
|
|
scope :for_associated_asset, lambda { |associated_asset| where(:associated_asset_type => associated_asset.class.to_s, :associated_asset_id => associated_asset.id) }
|
|
scope :active, lambda { where("content_tags.workflow_state <> 'deleted'").joins(:alignment) }
|
|
|
|
private
|
|
|
|
def infer_defaults
|
|
self.learning_outcome_id = self.alignment.learning_outcome_id
|
|
self.context_code = "#{self.context_type.underscore}_#{self.context_id}" rescue nil
|
|
self.original_score ||= self.score
|
|
self.original_possible ||= self.possible
|
|
self.original_mastery = self.mastery if self.original_mastery == nil
|
|
calculate_percent!
|
|
true
|
|
end
|
|
|
|
def calculate_by_scale(scale_data)
|
|
scale_percent = scale_data[:scale_percent]
|
|
alignment_mastery = scale_data[:alignment_mastery]
|
|
scale_points = (self.possible / scale_percent) - self.possible
|
|
scale_cutoff = self.possible - (self.possible * alignment_mastery)
|
|
percent_to_scale = (self.score + scale_cutoff) - self.possible
|
|
if percent_to_scale > 0 && scale_cutoff > 0
|
|
score_adjustment = (percent_to_scale / scale_cutoff) * scale_points
|
|
scaled_score = self.score + score_adjustment
|
|
(scaled_score / self.possible) * scale_percent
|
|
else
|
|
(self.score / self.possible) * scale_percent
|
|
end
|
|
end
|
|
|
|
def scale_params
|
|
parent_mastery = precise_mastery_percent
|
|
alignment_mastery = self.alignment.mastery_score
|
|
return unless parent_mastery && alignment_mastery
|
|
if parent_mastery > 0 && alignment_mastery > 0
|
|
{ scale_percent: parent_mastery / alignment_mastery,
|
|
alignment_mastery: alignment_mastery }
|
|
end
|
|
end
|
|
|
|
def parent_has_mastery?
|
|
parent_outcome&.mastery_points.to_f > 0
|
|
end
|
|
|
|
def parent_outcome
|
|
self.learning_outcome
|
|
end
|
|
|
|
def needs_scale?(scale_data)
|
|
scale_data && scale_data[:scale_percent] != 1.0
|
|
end
|
|
|
|
def precise_mastery_percent
|
|
# the outcome's mastery percent is rounded to 2 places. This is normally OK
|
|
# but for scaling it's too imprecise and can lead to inaccurate calculations
|
|
return unless parent_has_mastery? && parent_outcome.points_possible > 0
|
|
parent_outcome.mastery_points.to_f / parent_outcome.points_possible.to_f
|
|
end
|
|
end
|