# Redmine plugin for Document Management System "Features" # # Copyright (C) 2011 Vít Jonáš # # 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. class DmsfFolder < ActiveRecord::Base unloadable cattr_reader :invalid_characters @@invalid_characters = /\A[^\/\\\?":<>]*\z/ belongs_to :project belongs_to :folder, :class_name => "DmsfFolder", :foreign_key => "dmsf_folder_id" has_many :subfolders, :class_name => "DmsfFolder", :foreign_key => "dmsf_folder_id", :order => "title ASC" has_many :files, :class_name => "DmsfFile", :foreign_key => "dmsf_folder_id", :conditions => { :deleted => false } belongs_to :user validates_presence_of :title validates_uniqueness_of :title, :scope => [:dmsf_folder_id, :project_id] validates_format_of :title, :with => @@invalid_characters, :message => l(:error_contains_invalid_character) validate :check_cycle def check_cycle folders = [] self.subfolders.each {|f| folders.push(f)} folders.each do |folder| if folder == self.folder errors.add(:folder, l(:error_create_cycle_in_folder_dependency)) return false end folder.subfolders.each {|f| folders.push(f)} end return true end def self.project_root_folders(project) find(:all, :conditions => ["dmsf_folder_id is NULL and project_id = :project_id", {:project_id => project.id}], :order => "title ASC") end def delete return false if !self.subfolders.empty? || !self.files.empty? destroy end def dmsf_path folder = self path = [] while !folder.nil? path.unshift(folder) folder = folder.folder end path end def dmsf_path_str path = self.dmsf_path string_path = path.map { |element| element.title } string_path.join("/") end def notify? return true if self.notification return true if folder && folder.notify? return false end def notify_deactivate self.notification = false self.save! end def notify_activate self.notification = true self.save! end def self.directory_tree(project, current_folder = nil) tree = [[l(:link_documents), nil]] DmsfFolder.project_root_folders(project).each do |folder| unless folder == current_folder tree.push(["...#{folder.title}", folder.id]) directory_subtree(tree, folder, 2, current_folder) end end return tree end def deep_file_count file_count = self.files.length self.subfolders.each {|subfolder| file_count += subfolder.deep_file_count} file_count end private def self.directory_subtree(tree, folder, level, current_folder) folder.subfolders.each do |subfolder| unless subfolder == current_folder tree.push(["#{"..." * level}#{subfolder.title}", subfolder.id]) directory_subtree(tree, subfolder, level + 1, current_folder) end end end end