|
|
|
|
class HomeworkCommon < ApplicationRecord
|
|
|
|
|
# homework_type 1:普通作业 2:编程作业(弃用) 3:分组作业 4:实训作业
|
|
|
|
|
enum homework_type: { normal: 1, program: 2, group: 3, practice: 4 }, _suffix: true
|
|
|
|
|
has_many :homework_group_settings, dependent: :destroy
|
|
|
|
|
has_many :published_settings, -> { group_published }, class_name: "HomeworkGroupSetting"
|
|
|
|
|
has_many :student_works, -> { where(is_delete: 0) }
|
|
|
|
|
has_many :score_student_works, -> { where("is_delete = 0 and work_status != 0").order("work_score desc") }, class_name: "StudentWork"
|
|
|
|
|
has_one :homework_detail_manual, dependent: :destroy
|
|
|
|
|
has_many :student_works_scores
|
|
|
|
|
|
|
|
|
|
# 分组作业的设置
|
|
|
|
|
has_one :homework_detail_group, dependent: :destroy
|
|
|
|
|
|
|
|
|
|
belongs_to :course, counter_cache: true
|
|
|
|
|
belongs_to :homework_bank, optional: true
|
|
|
|
|
belongs_to :user
|
|
|
|
|
has_many :homework_challenge_settings, dependent: :destroy
|
|
|
|
|
|
|
|
|
|
has_one :homework_commons_shixun, dependent: :destroy
|
|
|
|
|
has_and_belongs_to_many :shixuns, through: :homework_commons_shixun
|
|
|
|
|
|
|
|
|
|
# attachtype: 1(描述的附件), 2(参考答案的附件)
|
|
|
|
|
has_many :attachments, as: :container, dependent: :destroy
|
|
|
|
|
|
|
|
|
|
# 作业的评论
|
|
|
|
|
has_many :journals_for_messages, as: :jour, dependent: :destroy
|
|
|
|
|
|
|
|
|
|
# 二级目录
|
|
|
|
|
belongs_to :course_second_category, optional: true
|
|
|
|
|
|
|
|
|
|
# 课堂动态
|
|
|
|
|
has_one :course_act, class_name: 'CourseActivity', as: :course_act, dependent: :destroy
|
|
|
|
|
has_many :tidings, as: :container, dependent: :destroy
|
|
|
|
|
# 实训作业的分班查重记录
|
|
|
|
|
has_many :homework_group_reviews, :dependent => :destroy
|
|
|
|
|
# 学生的查重情况
|
|
|
|
|
has_many :homework_review_results, :dependent => :destroy
|
|
|
|
|
|
|
|
|
|
validates :name, presence: true, length: { maximum: 60, too_long: "不能超过60个字符" }
|
|
|
|
|
validates :description, length: { maximum: 15000, too_long: "不能超过15000个字符" }
|
|
|
|
|
validates :explanation, length: { maximum: 5000, too_long: "不能超过5000个字符" }
|
|
|
|
|
validates :reference_answer, length: { maximum: 25000, too_long: "不能超过25000个字符" }
|
|
|
|
|
|
|
|
|
|
# after_update :update_activity
|
|
|
|
|
before_destroy :update_homework_bank_quotes
|
|
|
|
|
|
|
|
|
|
#1已发布的课堂作业
|
|
|
|
|
scope :homework_published, -> {where("homework_commons.publish_time IS NOT NULL AND homework_commons.publish_time <= ? ",Time.now)}
|
|
|
|
|
scope :published_no_end, -> {where("homework_commons.publish_time IS NOT NULL AND homework_commons.publish_time < ?
|
|
|
|
|
and homework_commons.end_time > ?", Time.now, Time.now)}
|
|
|
|
|
scope :homework_ended, -> {where("homework_commons.end_time IS NOT NULL AND homework_commons.end_time <= ? ",Time.now)}
|
|
|
|
|
scope :search_homework_type, lambda {|num| where(homework_type:num)}
|
|
|
|
|
scope :unified_setting, -> {where("unified_setting = ? ", 1)}
|
|
|
|
|
|
|
|
|
|
scope :normals, -> {where(homework_type: %i[normal]).order("position desc")}
|
|
|
|
|
scope :groups, -> {where(homework_type: %i[group]).order("position desc")}
|
|
|
|
|
scope :practices, -> {where(homework_type: %i[practice]).order("position desc")}
|
|
|
|
|
|
|
|
|
|
# 是否显示参考答案
|
|
|
|
|
def view_answer identity, user_id
|
|
|
|
|
identity < Course::STUDENT || (identity == Course::STUDENT && answer_public &&
|
|
|
|
|
self.end_or_late && self.user_work(user_id).try(:work_status).to_i > 0)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 申诉阶段
|
|
|
|
|
def appeal_duration
|
|
|
|
|
self.anonymous_appeal && [3, 4].include?(homework_detail_manual.comment_status)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业对应的子目录/父目录名称
|
|
|
|
|
def category_info
|
|
|
|
|
case self.homework_type
|
|
|
|
|
when 'normal'
|
|
|
|
|
{category_id: course.common_course_modules.first.try(:id), category_name: course.common_course_modules.first.try(:module_name), main: 1}
|
|
|
|
|
when 'group'
|
|
|
|
|
{category_id: course.group_course_modules.first.try(:id), category_name: course.group_course_modules.first.try(:module_name), main: 1}
|
|
|
|
|
when 'practice'
|
|
|
|
|
if self.course_second_category.present?
|
|
|
|
|
{category_id: self.course_second_category.try(:id), category_name: self.course_second_category.try(:name), main: 0}
|
|
|
|
|
else
|
|
|
|
|
{category_id: course.shixun_course_modules.take.try(:id), category_name: course.shixun_course_modules.take.try(:module_name), main: 1}
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 根据是否统一发布获取作业的作品列表
|
|
|
|
|
def all_works
|
|
|
|
|
student_works = self.unified_setting ? self.student_works :
|
|
|
|
|
self.student_works.joins("join course_members on student_works.user_id=course_members.user_id").
|
|
|
|
|
where(course_members: {course_group_id: self.published_settings.pluck(:course_group_id)})
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 分班权限的老师可见的作品列表
|
|
|
|
|
def teacher_works member
|
|
|
|
|
# member = course.course_member(user_id)
|
|
|
|
|
teacher_course_groups = member.try(:teacher_course_groups)
|
|
|
|
|
all_student_works = self.all_works
|
|
|
|
|
# 有分班权限的统计管理的分班且已发布的学生情况
|
|
|
|
|
if member.present? && teacher_course_groups.exists?
|
|
|
|
|
group_ids = teacher_course_groups.pluck(:course_group_id)
|
|
|
|
|
all_student_works = all_student_works.joins("join course_members on student_works.user_id=course_members.user_id").
|
|
|
|
|
where(course_members: {course_group_id: group_ids})
|
|
|
|
|
end
|
|
|
|
|
all_student_works
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def user_work user_id
|
|
|
|
|
work = StudentWork.find_by(homework_common_id: id, user_id: user_id) || StudentWork.create!(homework_common_id: id, user_id: user_id)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 是否在补交阶段内
|
|
|
|
|
def late_duration
|
|
|
|
|
homework_setting = self.homework_group_setting(User.current.id, true)
|
|
|
|
|
!course.is_end && self.publish_time && self.publish_time < Time.now && homework_setting.end_time &&
|
|
|
|
|
homework_setting.end_time < Time.now && self.allow_late && (self.late_time.nil? || self.late_time > Time.now)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业是否补交截止或者不允许补交且提交截止
|
|
|
|
|
def end_or_late
|
|
|
|
|
status = false
|
|
|
|
|
if self.course.is_end || (self.allow_late && self.late_time && self.late_time < Time.now)
|
|
|
|
|
status = true
|
|
|
|
|
elsif !self.allow_late
|
|
|
|
|
homework_setting = self.homework_group_setting(User.current.id, true)
|
|
|
|
|
status = homework_setting.end_time && homework_setting.end_time < Time.now
|
|
|
|
|
end
|
|
|
|
|
status
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业是否补交截止或者不允许补交且提交截止(不考虑分班)
|
|
|
|
|
def end_or_late_none_group
|
|
|
|
|
course.is_end || (allow_late && late_time < Time.now) || (!allow_late && end_time < Time.now)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业是否可以查重
|
|
|
|
|
def code_review
|
|
|
|
|
self.homework_type == 'practice' && self.publish_time.present? && self.publish_time < Time.now && self.homework_group_reviews.count == 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业查看最新成绩
|
|
|
|
|
def update_score identity
|
|
|
|
|
identity < Course::NORMAL && publish_time.present? && publish_time < Time.now && !end_or_late_none_group
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业能否立即发布
|
|
|
|
|
def publish_immediately charge_ids
|
|
|
|
|
homework_detail_manual.try(:comment_status) == 0 ||
|
|
|
|
|
homework_group_settings.select{|setting| charge_ids.include?(setting.course_group_id) &&
|
|
|
|
|
(setting.publish_time.nil? || setting.publish_time > Time.now)}.size > 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业能否立即截止
|
|
|
|
|
def end_immediately charge_ids
|
|
|
|
|
(unified_setting && homework_detail_manual.try(:comment_status) == 1 && end_time.present? && end_time > Time.now) ||
|
|
|
|
|
homework_group_settings.select{|setting| charge_ids.include?(setting.course_group_id) &&
|
|
|
|
|
!setting.publish_time.nil? && setting.publish_time < Time.now && setting.end_time > Time.now}.size > 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 学生是否提交了作品
|
|
|
|
|
def user_has_commit_work user_id
|
|
|
|
|
work = self.student_works.find_by_user_id(user_id)
|
|
|
|
|
work.present? && work.work_status != 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 是否有学生提交过作品
|
|
|
|
|
def has_commit_work
|
|
|
|
|
student_works.has_committed.count > 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 是否有学生关联了项目
|
|
|
|
|
def has_relate_project
|
|
|
|
|
homework_type == "group" && student_works.where("project_id != 0").count > 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业描述的附件
|
|
|
|
|
def des_attachments
|
|
|
|
|
attachments.where(attachtype: 1)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 参考答案的附件
|
|
|
|
|
def ref_attachments
|
|
|
|
|
attachments.where(attachtype: 2)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def update_activity
|
|
|
|
|
course_activity = CourseActivity.find_by_course_act_type_and_course_act_id(self.class.to_s, self.id)
|
|
|
|
|
|
|
|
|
|
course_activity.update_attributes(updated_at: Time.now) if course_activity
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
#删除时更新题库中的引用数
|
|
|
|
|
def update_homework_bank_quotes
|
|
|
|
|
old_bank = self.homework_bank
|
|
|
|
|
old_bank.update_attributes(quotes: (old_bank.quotes - 1) > 0 ? (old_bank.quotes - 1) : 0, homework_common_id: nil) if old_bank.present?
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 查重是否有新结果
|
|
|
|
|
def code_reviews_new_results?
|
|
|
|
|
self.homework_group_reviews.where(status: 0).count > 0
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 任务已评数 user_id: 对应老师的id
|
|
|
|
|
def comment_count(user_id)
|
|
|
|
|
course = self.course
|
|
|
|
|
# 如果有分班,则取此老师对应分班的已评数,没有分班,则取所有的
|
|
|
|
|
course_group_ids = course.group_course_power(user_id)
|
|
|
|
|
sql =
|
|
|
|
|
if course_group_ids.count > 0
|
|
|
|
|
%Q{
|
|
|
|
|
SELECT count(distinct sw.id) cnt FROM student_works sw
|
|
|
|
|
JOIN student_works_scores sws on sws.student_work_id = sw.id WHERE
|
|
|
|
|
reviewer_role IN(1,2) AND sw.id IN (SELECT id FROM student_works WHERE homework_common_id = #{self.id}
|
|
|
|
|
AND user_id IN (SELECT user_id FROM course_members WHERE role = 4 AND course_group_id
|
|
|
|
|
IN (#{course_group_ids.join(",")})
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
%Q{
|
|
|
|
|
SELECT COUNT(distinct sw.id) cnt FROM student_works sw
|
|
|
|
|
JOIN student_works_scores sws ON sw.id = sws.student_work_id
|
|
|
|
|
WHERE reviewer_role IN(1,2) AND sw.homework_common_id = #{self.id}
|
|
|
|
|
}
|
|
|
|
|
end
|
|
|
|
|
StudentWorksScore.find_by_sql(sql).first.try(:cnt).to_i
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 任务未评数
|
|
|
|
|
def uncomment_count(user_id)
|
|
|
|
|
user_ids = course.teacher_group_user_ids user_id
|
|
|
|
|
self.student_works.where(user_id: user_ids).count - self.comment_count(user_id)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作品未提交数
|
|
|
|
|
def unfinished_count member
|
|
|
|
|
self.teacher_works(member).unfinished.count
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 任务按时提交数
|
|
|
|
|
def finished_count member
|
|
|
|
|
self.teacher_works(member).finished.count
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def delay_finished_count member
|
|
|
|
|
self.teacher_works(member).delay_finished.count
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def compelete_status_count member, status
|
|
|
|
|
teacher_works(member).where(compelete_status: status).count
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 分组作业的最大分组id
|
|
|
|
|
def max_group_id
|
|
|
|
|
self.student_works.has_committed.maximum(:group_id).to_i + 1
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
# 作业的分班设置时间
|
|
|
|
|
def homework_group_setting user_id, current_user=false
|
|
|
|
|
if unified_setting
|
|
|
|
|
homework_setting = self
|
|
|
|
|
else
|
|
|
|
|
# 当前用户是从course_member中取,否则是从学生中取(双重身份的原因)
|
|
|
|
|
member = current_user ? course.course_member(user_id) : course.students.find_by(user_id: user_id)
|
|
|
|
|
group_setting = self.homework_group_settings.select{ |setting| setting.course_group_id == member.try(:course_group_id)}.first
|
|
|
|
|
homework_setting = group_setting.present? ? group_setting : self
|
|
|
|
|
end
|
|
|
|
|
homework_setting
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def min_group_publish_time
|
|
|
|
|
#HomeworkGroupSetting.where("homework_common_id = #{self.id} and publish_time is not null").pluck(:publish_time).min
|
|
|
|
|
# 可以使用includes
|
|
|
|
|
self.homework_group_settings.where("publish_time is not null").pluck(:publish_time).min
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def max_group_end_time
|
|
|
|
|
#HomeworkGroupSetting.where("homework_common_id = #{self.id} and end_time is not null").pluck(:end_time).max
|
|
|
|
|
# 可以使用includes
|
|
|
|
|
self.homework_group_settings.where("end_time is not null").pluck(:end_time).max
|
|
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def challenge_score challenge_id
|
|
|
|
|
homework_challenge_settings.find_by(challenge_id: challenge_id)&.score.to_f
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def create_homework_group_settings
|
|
|
|
|
if homework_group_settings.size != course.course_groups.size
|
|
|
|
|
course.course_groups.where.not(id: homework_group_settings.pluck(:course_group_id)).each do |group|
|
|
|
|
|
homework_group_settings << HomeworkGroupSetting.new(course_group_id: group.id, course_id: course.id,
|
|
|
|
|
publish_time: publish_time, end_time: end_time)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def update_homework_work_score
|
|
|
|
|
if unified_setting
|
|
|
|
|
works = student_works
|
|
|
|
|
user_ids = course.students.pluck(:user_id)
|
|
|
|
|
else
|
|
|
|
|
user_ids = course.students.where(course_group_id: published_settings.pluck(:course_group_id)).pluck(:user_id)
|
|
|
|
|
works = student_works.where(user_id: user_ids)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
works = works.includes(:challenge_work_scores)
|
|
|
|
|
|
|
|
|
|
challenge_settings = homework_challenge_settings
|
|
|
|
|
challenge_setting_ids = challenge_settings.pluck(:challenge_id)
|
|
|
|
|
myshixuns = Myshixun.where(shixun_id: homework_commons_shixun&.shixun_id, user_id: user_ids).includes(:games)
|
|
|
|
|
myshixuns.find_each(batch_size: 100) do |myshixun|
|
|
|
|
|
work = works.select{|work| work.user_id == myshixun.user_id}.first
|
|
|
|
|
if work && myshixun
|
|
|
|
|
games = myshixun.games.select{|game| challenge_setting_ids.include?(game.challenge_id)}
|
|
|
|
|
HomeworksService.new.update_myshixun_work_score work, myshixun, games, self, challenge_settings
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
HomeworksService.new.update_student_eff_score(self) if (allow_late && late_time < Time.now) ||
|
|
|
|
|
(!allow_late && end_time < Time.now)
|
|
|
|
|
update_attribute('calculation_time', Time.now)
|
|
|
|
|
end
|
|
|
|
|
end
|