341 lines
11 KiB
CoffeeScript
341 lines
11 KiB
CoffeeScript
###
|
|
# Copyright (C) 2014 Andrey Antukh <niwi@niwi.be>
|
|
# Copyright (C) 2014 Jesús Espino Garcia <jespinog@gmail.com>
|
|
# Copyright (C) 2014 David Barragán Merino <bameda@dbarragan.com>
|
|
#
|
|
# This program 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, either version 3 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 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/>.
|
|
#
|
|
# File: modules/admin/project-profile.coffee
|
|
###
|
|
|
|
taiga = @.taiga
|
|
|
|
mixOf = @.taiga.mixOf
|
|
trim = @.taiga.trim
|
|
toString = @.taiga.toString
|
|
joinStr = @.taiga.joinStr
|
|
groupBy = @.taiga.groupBy
|
|
bindOnce = @.taiga.bindOnce
|
|
|
|
module = angular.module("taigaAdmin")
|
|
|
|
#############################################################################
|
|
## Project values Controller
|
|
#############################################################################
|
|
|
|
class ProjectValuesController extends mixOf(taiga.Controller, taiga.PageMixin)
|
|
@.$inject = [
|
|
"$scope",
|
|
"$rootScope",
|
|
"$tgRepo",
|
|
"$tgConfirm",
|
|
"$tgResources",
|
|
"$routeParams",
|
|
"$q",
|
|
"$tgLocation",
|
|
"$tgNavUrls",
|
|
"$appTitle"
|
|
]
|
|
|
|
constructor: (@scope, @rootscope, @repo, @confirm, @rs, @params, @q, @location, @navUrls, @appTitle) ->
|
|
@scope.project = {}
|
|
|
|
promise = @.loadInitialData()
|
|
|
|
promise.then () =>
|
|
@appTitle.set("Project values - " + @scope.sectionName + " - " + @scope.project.name)
|
|
|
|
promise.then null, (xhr) =>
|
|
if xhr and xhr.status == 404
|
|
@location.path(@navUrls.resolve("not-found"))
|
|
@location.replace()
|
|
return @q.reject(xhr)
|
|
|
|
@scope.$on("admin:project-values:move", @.moveValue)
|
|
|
|
loadProject: ->
|
|
return @rs.projects.get(@scope.projectId).then (project) =>
|
|
@scope.project = project
|
|
@scope.$emit('project:loaded', project)
|
|
return project
|
|
|
|
loadValues: =>
|
|
return @rs[@scope.resource].listValues(@scope.projectId, @scope.type).then (values) =>
|
|
@scope.values = values
|
|
@scope.maxValueOrder = _.max(values, "order").order
|
|
|
|
loadInitialData: ->
|
|
promise = @repo.resolve({pslug: @params.pslug}).then (data) =>
|
|
@scope.projectId = data.project
|
|
return data
|
|
|
|
return promise.then( => @q.all([
|
|
@.loadProject(),
|
|
@.loadValues(),
|
|
]))
|
|
|
|
moveValue: (ctx, itemValue, itemIndex) =>
|
|
values = @scope.values
|
|
r = values.indexOf(itemValue)
|
|
values.splice(r, 1)
|
|
values.splice(itemIndex, 0, itemValue)
|
|
_.each values, (value, index) ->
|
|
value.order = index
|
|
|
|
@repo.saveAll(values)
|
|
|
|
module.controller("ProjectValuesController", ProjectValuesController)
|
|
|
|
|
|
#############################################################################
|
|
## Project values directive
|
|
#############################################################################
|
|
|
|
ProjectValuesDirective = ($log, $repo, $confirm, $location, animationFrame) ->
|
|
## Drag & Drop Link
|
|
|
|
linkDragAndDrop = ($scope, $el, $attrs) ->
|
|
oldParentScope = null
|
|
newParentScope = null
|
|
itemEl = null
|
|
tdom = $el.find(".sortable")
|
|
|
|
tdom.sortable({
|
|
handle: ".row.table-main.visualization",
|
|
dropOnEmpty: true
|
|
connectWith: ".project-values-body"
|
|
revert: 400
|
|
axis: "y"
|
|
})
|
|
|
|
tdom.on "sortstop", (event, ui) ->
|
|
itemEl = ui.item
|
|
itemValue = itemEl.scope().value
|
|
itemIndex = itemEl.index()
|
|
$scope.$broadcast("admin:project-values:move", itemValue, itemIndex)
|
|
|
|
$scope.$on "$destroy", ->
|
|
$el.off()
|
|
|
|
## Value Link
|
|
|
|
linkValue = ($scope, $el, $attrs) ->
|
|
$ctrl = $el.controller()
|
|
valueType = $attrs.type
|
|
|
|
initializeNewValue = ->
|
|
$scope.newValue = {
|
|
"name": ""
|
|
"is_closed": false
|
|
}
|
|
|
|
initializeNewValue()
|
|
|
|
goToBottomList = (focus = false) =>
|
|
table = $el.find(".table-main")
|
|
|
|
console.log(table.offset().top + table.height())
|
|
|
|
$(document.body).scrollTop(table.offset().top + table.height())
|
|
|
|
if focus
|
|
$(".new-value input").focus()
|
|
|
|
submit = =>
|
|
promise = $repo.save($scope.project)
|
|
promise.then ->
|
|
$confirm.notify("success")
|
|
|
|
promise.then null, (data) ->
|
|
$confirm.notify("error", data._error_message)
|
|
|
|
saveValue = (target)->
|
|
form = target.parents("form").checksley()
|
|
return if not form.validate()
|
|
|
|
value = target.scope().value
|
|
promise = $repo.save(value)
|
|
promise.then =>
|
|
row = target.parents(".row.table-main")
|
|
row.hide()
|
|
row.siblings(".visualization").css("display": "flex")
|
|
|
|
promise.then null, (data) ->
|
|
$confirm.notify("error")
|
|
form.setErrors(data)
|
|
|
|
cancel = (target) ->
|
|
row = target.parents(".row.table-main")
|
|
value = target.scope().value
|
|
$scope.$apply ->
|
|
row.hide()
|
|
value.revert()
|
|
row.siblings(".visualization").css("display": "flex")
|
|
|
|
$el.on "submit", "form", (event) ->
|
|
event.preventDefault()
|
|
submit()
|
|
|
|
$el.on "click", "form a.button-green", (event) ->
|
|
event.preventDefault()
|
|
submit()
|
|
|
|
$el.on "click", ".show-add-new", (event) ->
|
|
event.preventDefault()
|
|
$el.find(".new-value").css('display': 'flex')
|
|
|
|
goToBottomList(true)
|
|
|
|
$el.on "click", ".add-new", (event) ->
|
|
event.preventDefault()
|
|
form = $el.find(".new-value").parents("form").checksley()
|
|
return if not form.validate()
|
|
|
|
$scope.newValue.project = $scope.project.id
|
|
|
|
$scope.newValue.order = if $scope.maxValueOrder then $scope.maxValueOrder + 1 else 1
|
|
|
|
promise = $repo.create(valueType, $scope.newValue)
|
|
promise.then =>
|
|
$ctrl.loadValues().then ->
|
|
animationFrame.add () ->
|
|
goToBottomList()
|
|
|
|
$el.find(".new-value").hide()
|
|
initializeNewValue()
|
|
|
|
promise.then null, (data) ->
|
|
$confirm.notify("error")
|
|
form.setErrors(data)
|
|
|
|
$el.on "click", ".delete-new", (event) ->
|
|
event.preventDefault()
|
|
$el.find(".new-value").hide()
|
|
initializeNewValue()
|
|
|
|
$el.on "click", ".edit-value", (event) ->
|
|
event.preventDefault()
|
|
target = angular.element(event.currentTarget)
|
|
|
|
row = target.parents(".row.table-main")
|
|
row.hide()
|
|
editionRow = row.siblings(".edition")
|
|
editionRow.css("display": "flex")
|
|
editionRow.find('input:visible').first().focus().select()
|
|
|
|
$el.on "keyup", ".edition input", (event) ->
|
|
if event.keyCode == 13
|
|
target = angular.element(event.currentTarget)
|
|
saveValue(target)
|
|
else if event.keyCode == 27
|
|
target = angular.element(event.currentTarget)
|
|
cancel(target)
|
|
|
|
$el.on "click", ".save", (event) ->
|
|
event.preventDefault()
|
|
target = angular.element(event.currentTarget)
|
|
saveValue(target)
|
|
|
|
$el.on "click", ".cancel", (event) ->
|
|
event.preventDefault()
|
|
target = angular.element(event.currentTarget)
|
|
cancel(target)
|
|
|
|
$el.on "click", ".delete-value", (event) ->
|
|
event.preventDefault()
|
|
target = angular.element(event.currentTarget)
|
|
value = target.scope().value
|
|
choices = {}
|
|
_.each $scope.values, (option) ->
|
|
if value.id != option.id
|
|
choices[option.id] = option.name
|
|
|
|
#TODO: i18n
|
|
title = "Delete"
|
|
subtitle = value.name
|
|
if _.keys(choices).length == 0
|
|
return $confirm.error("You can't delete all values.")
|
|
|
|
return $confirm.askChoice(title, subtitle, choices).then (response) ->
|
|
onSucces = ->
|
|
$ctrl.loadValues().finally ->
|
|
response.finish()
|
|
onError = ->
|
|
$confirm.notify("error")
|
|
$repo.remove(value, {"moveTo": response.selected}).then(onSucces, onError)
|
|
|
|
link = ($scope, $el, $attrs) ->
|
|
linkDragAndDrop($scope, $el, $attrs)
|
|
linkValue($scope, $el, $attrs)
|
|
|
|
$scope.$on "$destroy", ->
|
|
$el.off()
|
|
|
|
return {link:link}
|
|
|
|
module.directive("tgProjectValues", ["$log", "$tgRepo", "$tgConfirm", "$tgLocation", "animationFrame",
|
|
ProjectValuesDirective])
|
|
|
|
|
|
#############################################################################
|
|
## Color selection directive
|
|
#############################################################################
|
|
|
|
ColorSelectionDirective = () ->
|
|
## Color selection Link
|
|
|
|
link = ($scope, $el, $attrs, $model) ->
|
|
$ctrl = $el.controller()
|
|
|
|
$scope.$watch $attrs.ngModel, (element) ->
|
|
$scope.color = element.color
|
|
|
|
$el.on "click", ".current-color", (event) ->
|
|
# Showing the color selector
|
|
event.preventDefault()
|
|
event.stopPropagation()
|
|
target = angular.element(event.currentTarget)
|
|
$el.find(".select-color").hide()
|
|
target.siblings(".select-color").show()
|
|
# Hide when click outside
|
|
body = angular.element("body")
|
|
body.on "click", (event) =>
|
|
if angular.element(event.target).parent(".select-color").length == 0
|
|
$el.find(".select-color").hide()
|
|
body.unbind("click")
|
|
|
|
$el.on "click", ".select-color .color", (event) ->
|
|
# Selecting one color on color selector
|
|
event.preventDefault()
|
|
target = angular.element(event.currentTarget)
|
|
$scope.$apply ->
|
|
$model.$modelValue.color = target.data("color")
|
|
$el.find(".select-color").hide()
|
|
|
|
$el.on "click", ".select-color .selected-color", (event) ->
|
|
event.preventDefault()
|
|
$scope.$apply ->
|
|
$model.$modelValue.color = $scope.color
|
|
$el.find(".select-color").hide()
|
|
|
|
$scope.$on "$destroy", ->
|
|
$el.off()
|
|
|
|
return {
|
|
link: link
|
|
require:"ngModel"
|
|
}
|
|
|
|
module.directive("tgColorSelection", ColorSelectionDirective)
|