Kanban board (#8346)
Co-authored-by: 6543 <6543@obermui.de> Co-authored-by: jaqra <48099350+jaqra@users.noreply.github.com> Co-authored-by: Kerry <flatline-studios@users.noreply.github.com> Co-authored-by: Jaqra <jaqra@hotmail.com> Co-authored-by: Kyle Evans <kevans91@users.noreply.github.com> Co-authored-by: Tsakiridis Ilias <TsakiDev@users.noreply.github.com> Co-authored-by: Ilias Tsakiridis <ilias.tsakiridis@outlook.com> Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com> Co-authored-by: silverwind <me@silverwind.io> Co-authored-by: zeripath <art27@cantab.net> Co-authored-by: techknowlogick <techknowlogick@gitea.io>tokarchuk/v1.17
parent
d285b5d35a
commit
4027c5dd7c
@ -0,0 +1 @@ |
|||||||
|
ref: refs/heads/master |
@ -0,0 +1,4 @@ |
|||||||
|
[core] |
||||||
|
repositoryformatversion = 0 |
||||||
|
filemode = true |
||||||
|
bare = true |
@ -0,0 +1 @@ |
|||||||
|
Unnamed repository; edit this file 'description' to name the repository. |
@ -0,0 +1,15 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to check the commit log message taken by |
||||||
|
# applypatch from an e-mail message. |
||||||
|
# |
||||||
|
# The hook should exit with non-zero status after issuing an |
||||||
|
# appropriate message if it wants to stop the commit. The hook is |
||||||
|
# allowed to edit the commit message file. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "applypatch-msg". |
||||||
|
|
||||||
|
. git-sh-setup |
||||||
|
commitmsg="$(git rev-parse --git-path hooks/commit-msg)" |
||||||
|
test -x "$commitmsg" && exec "$commitmsg" ${1+"$@"} |
||||||
|
: |
@ -0,0 +1,24 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to check the commit log message. |
||||||
|
# Called by "git commit" with one argument, the name of the file |
||||||
|
# that has the commit message. The hook should exit with non-zero |
||||||
|
# status after issuing an appropriate message if it wants to stop the |
||||||
|
# commit. The hook is allowed to edit the commit message file. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "commit-msg". |
||||||
|
|
||||||
|
# Uncomment the below to add a Signed-off-by line to the message. |
||||||
|
# Doing this in a hook is a bad idea in general, but the prepare-commit-msg |
||||||
|
# hook is more suited to it. |
||||||
|
# |
||||||
|
# SOB=$(git var GIT_AUTHOR_IDENT | sed -n 's/^\(.*>\).*$/Signed-off-by: \1/p') |
||||||
|
# grep -qs "^$SOB" "$1" || echo "$SOB" >> "$1" |
||||||
|
|
||||||
|
# This example catches duplicate Signed-off-by lines. |
||||||
|
|
||||||
|
test "" = "$(grep '^Signed-off-by: ' "$1" | |
||||||
|
sort | uniq -c | sed -e '/^[ ]*1[ ]/d')" || { |
||||||
|
echo >&2 Duplicate Signed-off-by lines. |
||||||
|
exit 1 |
||||||
|
} |
@ -0,0 +1,173 @@ |
|||||||
|
#!/usr/bin/perl |
||||||
|
|
||||||
|
use strict; |
||||||
|
use warnings; |
||||||
|
use IPC::Open2; |
||||||
|
|
||||||
|
# An example hook script to integrate Watchman |
||||||
|
# (https://facebook.github.io/watchman/) with git to speed up detecting |
||||||
|
# new and modified files. |
||||||
|
# |
||||||
|
# The hook is passed a version (currently 2) and last update token |
||||||
|
# formatted as a string and outputs to stdout a new update token and |
||||||
|
# all files that have been modified since the update token. Paths must |
||||||
|
# be relative to the root of the working tree and separated by a single NUL. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "query-watchman" and set |
||||||
|
# 'git config core.fsmonitor .git/hooks/query-watchman' |
||||||
|
# |
||||||
|
my ($version, $last_update_token) = @ARGV; |
||||||
|
|
||||||
|
# Uncomment for debugging |
||||||
|
# print STDERR "$0 $version $last_update_token\n"; |
||||||
|
|
||||||
|
# Check the hook interface version |
||||||
|
if ($version ne 2) { |
||||||
|
die "Unsupported query-fsmonitor hook version '$version'.\n" . |
||||||
|
"Falling back to scanning...\n"; |
||||||
|
} |
||||||
|
|
||||||
|
my $git_work_tree = get_working_dir(); |
||||||
|
|
||||||
|
my $retry = 1; |
||||||
|
|
||||||
|
my $json_pkg; |
||||||
|
eval { |
||||||
|
require JSON::XS; |
||||||
|
$json_pkg = "JSON::XS"; |
||||||
|
1; |
||||||
|
} or do { |
||||||
|
require JSON::PP; |
||||||
|
$json_pkg = "JSON::PP"; |
||||||
|
}; |
||||||
|
|
||||||
|
launch_watchman(); |
||||||
|
|
||||||
|
sub launch_watchman { |
||||||
|
my $o = watchman_query(); |
||||||
|
if (is_work_tree_watched($o)) { |
||||||
|
output_result($o->{clock}, @{$o->{files}}); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
sub output_result { |
||||||
|
my ($clockid, @files) = @_; |
||||||
|
|
||||||
|
# Uncomment for debugging watchman output |
||||||
|
# open (my $fh, ">", ".git/watchman-output.out"); |
||||||
|
# binmode $fh, ":utf8"; |
||||||
|
# print $fh "$clockid\n@files\n"; |
||||||
|
# close $fh; |
||||||
|
|
||||||
|
binmode STDOUT, ":utf8"; |
||||||
|
print $clockid; |
||||||
|
print "\0"; |
||||||
|
local $, = "\0"; |
||||||
|
print @files; |
||||||
|
} |
||||||
|
|
||||||
|
sub watchman_clock { |
||||||
|
my $response = qx/watchman clock "$git_work_tree"/; |
||||||
|
die "Failed to get clock id on '$git_work_tree'.\n" . |
||||||
|
"Falling back to scanning...\n" if $? != 0; |
||||||
|
|
||||||
|
return $json_pkg->new->utf8->decode($response); |
||||||
|
} |
||||||
|
|
||||||
|
sub watchman_query { |
||||||
|
my $pid = open2(\*CHLD_OUT, \*CHLD_IN, 'watchman -j --no-pretty') |
||||||
|
or die "open2() failed: $!\n" . |
||||||
|
"Falling back to scanning...\n"; |
||||||
|
|
||||||
|
# In the query expression below we're asking for names of files that |
||||||
|
# changed since $last_update_token but not from the .git folder. |
||||||
|
# |
||||||
|
# To accomplish this, we're using the "since" generator to use the |
||||||
|
# recency index to select candidate nodes and "fields" to limit the |
||||||
|
# output to file names only. Then we're using the "expression" term to |
||||||
|
# further constrain the results. |
||||||
|
if (substr($last_update_token, 0, 1) eq "c") { |
||||||
|
$last_update_token = "\"$last_update_token\""; |
||||||
|
} |
||||||
|
my $query = <<" END"; |
||||||
|
["query", "$git_work_tree", { |
||||||
|
"since": $last_update_token, |
||||||
|
"fields": ["name"], |
||||||
|
"expression": ["not", ["dirname", ".git"]] |
||||||
|
}] |
||||||
|
END |
||||||
|
|
||||||
|
# Uncomment for debugging the watchman query |
||||||
|
# open (my $fh, ">", ".git/watchman-query.json"); |
||||||
|
# print $fh $query; |
||||||
|
# close $fh; |
||||||
|
|
||||||
|
print CHLD_IN $query; |
||||||
|
close CHLD_IN; |
||||||
|
my $response = do {local $/; <CHLD_OUT>}; |
||||||
|
|
||||||
|
# Uncomment for debugging the watch response |
||||||
|
# open ($fh, ">", ".git/watchman-response.json"); |
||||||
|
# print $fh $response; |
||||||
|
# close $fh; |
||||||
|
|
||||||
|
die "Watchman: command returned no output.\n" . |
||||||
|
"Falling back to scanning...\n" if $response eq ""; |
||||||
|
die "Watchman: command returned invalid output: $response\n" . |
||||||
|
"Falling back to scanning...\n" unless $response =~ /^\{/; |
||||||
|
|
||||||
|
return $json_pkg->new->utf8->decode($response); |
||||||
|
} |
||||||
|
|
||||||
|
sub is_work_tree_watched { |
||||||
|
my ($output) = @_; |
||||||
|
my $error = $output->{error}; |
||||||
|
if ($retry > 0 and $error and $error =~ m/unable to resolve root .* directory (.*) is not watched/) { |
||||||
|
$retry--; |
||||||
|
my $response = qx/watchman watch "$git_work_tree"/; |
||||||
|
die "Failed to make watchman watch '$git_work_tree'.\n" . |
||||||
|
"Falling back to scanning...\n" if $? != 0; |
||||||
|
$output = $json_pkg->new->utf8->decode($response); |
||||||
|
$error = $output->{error}; |
||||||
|
die "Watchman: $error.\n" . |
||||||
|
"Falling back to scanning...\n" if $error; |
||||||
|
|
||||||
|
# Uncomment for debugging watchman output |
||||||
|
# open (my $fh, ">", ".git/watchman-output.out"); |
||||||
|
# close $fh; |
||||||
|
|
||||||
|
# Watchman will always return all files on the first query so |
||||||
|
# return the fast "everything is dirty" flag to git and do the |
||||||
|
# Watchman query just to get it over with now so we won't pay |
||||||
|
# the cost in git to look up each individual file. |
||||||
|
my $o = watchman_clock(); |
||||||
|
$error = $output->{error}; |
||||||
|
|
||||||
|
die "Watchman: $error.\n" . |
||||||
|
"Falling back to scanning...\n" if $error; |
||||||
|
|
||||||
|
output_result($o->{clock}, ("/")); |
||||||
|
$last_update_token = $o->{clock}; |
||||||
|
|
||||||
|
eval { launch_watchman() }; |
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
die "Watchman: $error.\n" . |
||||||
|
"Falling back to scanning...\n" if $error; |
||||||
|
|
||||||
|
return 1; |
||||||
|
} |
||||||
|
|
||||||
|
sub get_working_dir { |
||||||
|
my $working_dir; |
||||||
|
if ($^O =~ 'msys' || $^O =~ 'cygwin') { |
||||||
|
$working_dir = Win32::GetCwd(); |
||||||
|
$working_dir =~ tr/\\/\//; |
||||||
|
} else { |
||||||
|
require Cwd; |
||||||
|
$working_dir = Cwd::cwd(); |
||||||
|
} |
||||||
|
|
||||||
|
return $working_dir; |
||||||
|
} |
@ -0,0 +1,8 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to prepare a packed repository for use over |
||||||
|
# dumb transports. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "post-update". |
||||||
|
|
||||||
|
exec git update-server-info |
@ -0,0 +1,14 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to verify what is about to be committed |
||||||
|
# by applypatch from an e-mail message. |
||||||
|
# |
||||||
|
# The hook should exit with non-zero status after issuing an |
||||||
|
# appropriate message if it wants to stop the commit. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "pre-applypatch". |
||||||
|
|
||||||
|
. git-sh-setup |
||||||
|
precommit="$(git rev-parse --git-path hooks/pre-commit)" |
||||||
|
test -x "$precommit" && exec "$precommit" ${1+"$@"} |
||||||
|
: |
@ -0,0 +1,49 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to verify what is about to be committed. |
||||||
|
# Called by "git commit" with no arguments. The hook should |
||||||
|
# exit with non-zero status after issuing an appropriate message if |
||||||
|
# it wants to stop the commit. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "pre-commit". |
||||||
|
|
||||||
|
if git rev-parse --verify HEAD >/dev/null 2>&1 |
||||||
|
then |
||||||
|
against=HEAD |
||||||
|
else |
||||||
|
# Initial commit: diff against an empty tree object |
||||||
|
against=$(git hash-object -t tree /dev/null) |
||||||
|
fi |
||||||
|
|
||||||
|
# If you want to allow non-ASCII filenames set this variable to true. |
||||||
|
allownonascii=$(git config --type=bool hooks.allownonascii) |
||||||
|
|
||||||
|
# Redirect output to stderr. |
||||||
|
exec 1>&2 |
||||||
|
|
||||||
|
# Cross platform projects tend to avoid non-ASCII filenames; prevent |
||||||
|
# them from being added to the repository. We exploit the fact that the |
||||||
|
# printable range starts at the space character and ends with tilde. |
||||||
|
if [ "$allownonascii" != "true" ] && |
||||||
|
# Note that the use of brackets around a tr range is ok here, (it's |
||||||
|
# even required, for portability to Solaris 10's /usr/bin/tr), since |
||||||
|
# the square bracket bytes happen to fall in the designated range. |
||||||
|
test $(git diff --cached --name-only --diff-filter=A -z $against | |
||||||
|
LC_ALL=C tr -d '[ -~]\0' | wc -c) != 0 |
||||||
|
then |
||||||
|
cat <<\EOF |
||||||
|
Error: Attempt to add a non-ASCII file name. |
||||||
|
|
||||||
|
This can cause problems if you want to work with people on other platforms. |
||||||
|
|
||||||
|
To be portable it is advisable to rename the file. |
||||||
|
|
||||||
|
If you know what you are doing you can disable this check using: |
||||||
|
|
||||||
|
git config hooks.allownonascii true |
||||||
|
EOF |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
|
||||||
|
# If there are whitespace errors, print the offending file names and fail. |
||||||
|
exec git diff-index --check --cached $against -- |
@ -0,0 +1,13 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to verify what is about to be committed. |
||||||
|
# Called by "git merge" with no arguments. The hook should |
||||||
|
# exit with non-zero status after issuing an appropriate message to |
||||||
|
# stderr if it wants to stop the merge commit. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "pre-merge-commit". |
||||||
|
|
||||||
|
. git-sh-setup |
||||||
|
test -x "$GIT_DIR/hooks/pre-commit" && |
||||||
|
exec "$GIT_DIR/hooks/pre-commit" |
||||||
|
: |
@ -0,0 +1,53 @@ |
|||||||
|
#!/bin/sh |
||||||
|
|
||||||
|
# An example hook script to verify what is about to be pushed. Called by "git |
||||||
|
# push" after it has checked the remote status, but before anything has been |
||||||
|
# pushed. If this script exits with a non-zero status nothing will be pushed. |
||||||
|
# |
||||||
|
# This hook is called with the following parameters: |
||||||
|
# |
||||||
|
# $1 -- Name of the remote to which the push is being done |
||||||
|
# $2 -- URL to which the push is being done |
||||||
|
# |
||||||
|
# If pushing without using a named remote those arguments will be equal. |
||||||
|
# |
||||||
|
# Information about the commits which are being pushed is supplied as lines to |
||||||
|
# the standard input in the form: |
||||||
|
# |
||||||
|
# <local ref> <local sha1> <remote ref> <remote sha1> |
||||||
|
# |
||||||
|
# This sample shows how to prevent push of commits where the log message starts |
||||||
|
# with "WIP" (work in progress). |
||||||
|
|
||||||
|
remote="$1" |
||||||
|
url="$2" |
||||||
|
|
||||||
|
z40=0000000000000000000000000000000000000000 |
||||||
|
|
||||||
|
while read local_ref local_sha remote_ref remote_sha |
||||||
|
do |
||||||
|
if [ "$local_sha" = $z40 ] |
||||||
|
then |
||||||
|
# Handle delete |
||||||
|
: |
||||||
|
else |
||||||
|
if [ "$remote_sha" = $z40 ] |
||||||
|
then |
||||||
|
# New branch, examine all commits |
||||||
|
range="$local_sha" |
||||||
|
else |
||||||
|
# Update to existing branch, examine new commits |
||||||
|
range="$remote_sha..$local_sha" |
||||||
|
fi |
||||||
|
|
||||||
|
# Check for WIP commit |
||||||
|
commit=`git rev-list -n 1 --grep '^WIP' "$range"` |
||||||
|
if [ -n "$commit" ] |
||||||
|
then |
||||||
|
echo >&2 "Found WIP commit in $local_ref, not pushing" |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
fi |
||||||
|
done |
||||||
|
|
||||||
|
exit 0 |
@ -0,0 +1,169 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# Copyright (c) 2006, 2008 Junio C Hamano |
||||||
|
# |
||||||
|
# The "pre-rebase" hook is run just before "git rebase" starts doing |
||||||
|
# its job, and can prevent the command from running by exiting with |
||||||
|
# non-zero status. |
||||||
|
# |
||||||
|
# The hook is called with the following parameters: |
||||||
|
# |
||||||
|
# $1 -- the upstream the series was forked from. |
||||||
|
# $2 -- the branch being rebased (or empty when rebasing the current branch). |
||||||
|
# |
||||||
|
# This sample shows how to prevent topic branches that are already |
||||||
|
# merged to 'next' branch from getting rebased, because allowing it |
||||||
|
# would result in rebasing already published history. |
||||||
|
|
||||||
|
publish=next |
||||||
|
basebranch="$1" |
||||||
|
if test "$#" = 2 |
||||||
|
then |
||||||
|
topic="refs/heads/$2" |
||||||
|
else |
||||||
|
topic=`git symbolic-ref HEAD` || |
||||||
|
exit 0 ;# we do not interrupt rebasing detached HEAD |
||||||
|
fi |
||||||
|
|
||||||
|
case "$topic" in |
||||||
|
refs/heads/??/*) |
||||||
|
;; |
||||||
|
*) |
||||||
|
exit 0 ;# we do not interrupt others. |
||||||
|
;; |
||||||
|
esac |
||||||
|
|
||||||
|
# Now we are dealing with a topic branch being rebased |
||||||
|
# on top of master. Is it OK to rebase it? |
||||||
|
|
||||||
|
# Does the topic really exist? |
||||||
|
git show-ref -q "$topic" || { |
||||||
|
echo >&2 "No such branch $topic" |
||||||
|
exit 1 |
||||||
|
} |
||||||
|
|
||||||
|
# Is topic fully merged to master? |
||||||
|
not_in_master=`git rev-list --pretty=oneline ^master "$topic"` |
||||||
|
if test -z "$not_in_master" |
||||||
|
then |
||||||
|
echo >&2 "$topic is fully merged to master; better remove it." |
||||||
|
exit 1 ;# we could allow it, but there is no point. |
||||||
|
fi |
||||||
|
|
||||||
|
# Is topic ever merged to next? If so you should not be rebasing it. |
||||||
|
only_next_1=`git rev-list ^master "^$topic" ${publish} | sort` |
||||||
|
only_next_2=`git rev-list ^master ${publish} | sort` |
||||||
|
if test "$only_next_1" = "$only_next_2" |
||||||
|
then |
||||||
|
not_in_topic=`git rev-list "^$topic" master` |
||||||
|
if test -z "$not_in_topic" |
||||||
|
then |
||||||
|
echo >&2 "$topic is already up to date with master" |
||||||
|
exit 1 ;# we could allow it, but there is no point. |
||||||
|
else |
||||||
|
exit 0 |
||||||
|
fi |
||||||
|
else |
||||||
|
not_in_next=`git rev-list --pretty=oneline ^${publish} "$topic"` |
||||||
|
/usr/bin/perl -e ' |
||||||
|
my $topic = $ARGV[0]; |
||||||
|
my $msg = "* $topic has commits already merged to public branch:\n"; |
||||||
|
my (%not_in_next) = map { |
||||||
|
/^([0-9a-f]+) /; |
||||||
|
($1 => 1); |
||||||
|
} split(/\n/, $ARGV[1]); |
||||||
|
for my $elem (map { |
||||||
|
/^([0-9a-f]+) (.*)$/; |
||||||
|
[$1 => $2]; |
||||||
|
} split(/\n/, $ARGV[2])) { |
||||||
|
if (!exists $not_in_next{$elem->[0]}) { |
||||||
|
if ($msg) { |
||||||
|
print STDERR $msg; |
||||||
|
undef $msg; |
||||||
|
} |
||||||
|
print STDERR " $elem->[1]\n"; |
||||||
|
} |
||||||
|
} |
||||||
|
' "$topic" "$not_in_next" "$not_in_master" |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
|
||||||
|
<<\DOC_END |
||||||
|
|
||||||
|
This sample hook safeguards topic branches that have been |
||||||
|
published from being rewound. |
||||||
|
|
||||||
|
The workflow assumed here is: |
||||||
|
|
||||||
|
* Once a topic branch forks from "master", "master" is never |
||||||
|
merged into it again (either directly or indirectly). |
||||||
|
|
||||||
|
* Once a topic branch is fully cooked and merged into "master", |
||||||
|
it is deleted. If you need to build on top of it to correct |
||||||
|
earlier mistakes, a new topic branch is created by forking at |
||||||
|
the tip of the "master". This is not strictly necessary, but |
||||||
|
it makes it easier to keep your history simple. |
||||||
|
|
||||||
|
* Whenever you need to test or publish your changes to topic |
||||||
|
branches, merge them into "next" branch. |
||||||
|
|
||||||
|
The script, being an example, hardcodes the publish branch name |
||||||
|
to be "next", but it is trivial to make it configurable via |
||||||
|
$GIT_DIR/config mechanism. |
||||||
|
|
||||||
|
With this workflow, you would want to know: |
||||||
|
|
||||||
|
(1) ... if a topic branch has ever been merged to "next". Young |
||||||
|
topic branches can have stupid mistakes you would rather |
||||||
|
clean up before publishing, and things that have not been |
||||||
|
merged into other branches can be easily rebased without |
||||||
|
affecting other people. But once it is published, you would |
||||||
|
not want to rewind it. |
||||||
|
|
||||||
|
(2) ... if a topic branch has been fully merged to "master". |
||||||
|
Then you can delete it. More importantly, you should not |
||||||
|
build on top of it -- other people may already want to |
||||||
|
change things related to the topic as patches against your |
||||||
|
"master", so if you need further changes, it is better to |
||||||
|
fork the topic (perhaps with the same name) afresh from the |
||||||
|
tip of "master". |
||||||
|
|
||||||
|
Let's look at this example: |
||||||
|
|
||||||
|
o---o---o---o---o---o---o---o---o---o "next" |
||||||
|
/ / / / |
||||||
|
/ a---a---b A / / |
||||||
|
/ / / / |
||||||
|
/ / c---c---c---c B / |
||||||
|
/ / / \ / |
||||||
|
/ / / b---b C \ / |
||||||
|
/ / / / \ / |
||||||
|
---o---o---o---o---o---o---o---o---o---o---o "master" |
||||||
|
|
||||||
|
|
||||||
|
A, B and C are topic branches. |
||||||
|
|
||||||
|
* A has one fix since it was merged up to "next". |
||||||
|
|
||||||
|
* B has finished. It has been fully merged up to "master" and "next", |
||||||
|
and is ready to be deleted. |
||||||
|
|
||||||
|
* C has not merged to "next" at all. |
||||||
|
|
||||||
|
We would want to allow C to be rebased, refuse A, and encourage |
||||||
|
B to be deleted. |
||||||
|
|
||||||
|
To compute (1): |
||||||
|
|
||||||
|
git rev-list ^master ^topic next |
||||||
|
git rev-list ^master next |
||||||
|
|
||||||
|
if these match, topic has not merged in next at all. |
||||||
|
|
||||||
|
To compute (2): |
||||||
|
|
||||||
|
git rev-list master..topic |
||||||
|
|
||||||
|
if this is empty, it is fully merged to "master". |
||||||
|
|
||||||
|
DOC_END |
@ -0,0 +1,24 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to make use of push options. |
||||||
|
# The example simply echoes all push options that start with 'echoback=' |
||||||
|
# and rejects all pushes when the "reject" push option is used. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "pre-receive". |
||||||
|
|
||||||
|
if test -n "$GIT_PUSH_OPTION_COUNT" |
||||||
|
then |
||||||
|
i=0 |
||||||
|
while test "$i" -lt "$GIT_PUSH_OPTION_COUNT" |
||||||
|
do |
||||||
|
eval "value=\$GIT_PUSH_OPTION_$i" |
||||||
|
case "$value" in |
||||||
|
echoback=*) |
||||||
|
echo "echo from the pre-receive-hook: ${value#*=}" >&2 |
||||||
|
;; |
||||||
|
reject) |
||||||
|
exit 1 |
||||||
|
esac |
||||||
|
i=$((i + 1)) |
||||||
|
done |
||||||
|
fi |
@ -0,0 +1,42 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to prepare the commit log message. |
||||||
|
# Called by "git commit" with the name of the file that has the |
||||||
|
# commit message, followed by the description of the commit |
||||||
|
# message's source. The hook's purpose is to edit the commit |
||||||
|
# message file. If the hook fails with a non-zero status, |
||||||
|
# the commit is aborted. |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "prepare-commit-msg". |
||||||
|
|
||||||
|
# This hook includes three examples. The first one removes the |
||||||
|
# "# Please enter the commit message..." help message. |
||||||
|
# |
||||||
|
# The second includes the output of "git diff --name-status -r" |
||||||
|
# into the message, just before the "git status" output. It is |
||||||
|
# commented because it doesn't cope with --amend or with squashed |
||||||
|
# commits. |
||||||
|
# |
||||||
|
# The third example adds a Signed-off-by line to the message, that can |
||||||
|
# still be edited. This is rarely a good idea. |
||||||
|
|
||||||
|
COMMIT_MSG_FILE=$1 |
||||||
|
COMMIT_SOURCE=$2 |
||||||
|
SHA1=$3 |
||||||
|
|
||||||
|
/usr/bin/perl -i.bak -ne 'print unless(m/^. Please enter the commit message/..m/^#$/)' "$COMMIT_MSG_FILE" |
||||||
|
|
||||||
|
# case "$COMMIT_SOURCE,$SHA1" in |
||||||
|
# ,|template,) |
||||||
|
# /usr/bin/perl -i.bak -pe ' |
||||||
|
# print "\n" . `git diff --cached --name-status -r` |
||||||
|
# if /^#/ && $first++ == 0' "$COMMIT_MSG_FILE" ;; |
||||||
|
# *) ;; |
||||||
|
# esac |
||||||
|
|
||||||
|
# SOB=$(git var GIT_COMMITTER_IDENT | sed -n 's/^\(.*>\).*$/Signed-off-by: \1/p') |
||||||
|
# git interpret-trailers --in-place --trailer "$SOB" "$COMMIT_MSG_FILE" |
||||||
|
# if test -z "$COMMIT_SOURCE" |
||||||
|
# then |
||||||
|
# /usr/bin/perl -i.bak -pe 'print "\n" if !$first_line++' "$COMMIT_MSG_FILE" |
||||||
|
# fi |
@ -0,0 +1,128 @@ |
|||||||
|
#!/bin/sh |
||||||
|
# |
||||||
|
# An example hook script to block unannotated tags from entering. |
||||||
|
# Called by "git receive-pack" with arguments: refname sha1-old sha1-new |
||||||
|
# |
||||||
|
# To enable this hook, rename this file to "update". |
||||||
|
# |
||||||
|
# Config |
||||||
|
# ------ |
||||||
|
# hooks.allowunannotated |
||||||
|
# This boolean sets whether unannotated tags will be allowed into the |
||||||
|
# repository. By default they won't be. |
||||||
|
# hooks.allowdeletetag |
||||||
|
# This boolean sets whether deleting tags will be allowed in the |
||||||
|
# repository. By default they won't be. |
||||||
|
# hooks.allowmodifytag |
||||||
|
# This boolean sets whether a tag may be modified after creation. By default |
||||||
|
# it won't be. |
||||||
|
# hooks.allowdeletebranch |
||||||
|
# This boolean sets whether deleting branches will be allowed in the |
||||||
|
# repository. By default they won't be. |
||||||
|
# hooks.denycreatebranch |
||||||
|
# This boolean sets whether remotely creating branches will be denied |
||||||
|
# in the repository. By default this is allowed. |
||||||
|
# |
||||||
|
|
||||||
|
# --- Command line |
||||||
|
refname="$1" |
||||||
|
oldrev="$2" |
||||||
|
newrev="$3" |
||||||
|
|
||||||
|
# --- Safety check |
||||||
|
if [ -z "$GIT_DIR" ]; then |
||||||
|
echo "Don't run this script from the command line." >&2 |
||||||
|
echo " (if you want, you could supply GIT_DIR then run" >&2 |
||||||
|
echo " $0 <ref> <oldrev> <newrev>)" >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
|
||||||
|
if [ -z "$refname" -o -z "$oldrev" -o -z "$newrev" ]; then |
||||||
|
echo "usage: $0 <ref> <oldrev> <newrev>" >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
|
||||||
|
# --- Config |
||||||
|
allowunannotated=$(git config --type=bool hooks.allowunannotated) |
||||||
|
allowdeletebranch=$(git config --type=bool hooks.allowdeletebranch) |
||||||
|
denycreatebranch=$(git config --type=bool hooks.denycreatebranch) |
||||||
|
allowdeletetag=$(git config --type=bool hooks.allowdeletetag) |
||||||
|
allowmodifytag=$(git config --type=bool hooks.allowmodifytag) |
||||||
|
|
||||||
|
# check for no description |
||||||
|
projectdesc=$(sed -e '1q' "$GIT_DIR/description") |
||||||
|
case "$projectdesc" in |
||||||
|
"Unnamed repository"* | "") |
||||||
|
echo "*** Project description file hasn't been set" >&2 |
||||||
|
exit 1 |
||||||
|
;; |
||||||
|
esac |
||||||
|
|
||||||
|
# --- Check types |
||||||
|
# if $newrev is 0000...0000, it's a commit to delete a ref. |
||||||
|
zero="0000000000000000000000000000000000000000" |
||||||
|
if [ "$newrev" = "$zero" ]; then |
||||||
|
newrev_type=delete |
||||||
|
else |
||||||
|
newrev_type=$(git cat-file -t $newrev) |
||||||
|
fi |
||||||
|
|
||||||
|
case "$refname","$newrev_type" in |
||||||
|
refs/tags/*,commit) |
||||||
|
# un-annotated tag |
||||||
|
short_refname=${refname##refs/tags/} |
||||||
|
if [ "$allowunannotated" != "true" ]; then |
||||||
|
echo "*** The un-annotated tag, $short_refname, is not allowed in this repository" >&2 |
||||||
|
echo "*** Use 'git tag [ -a | -s ]' for tags you want to propagate." >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
;; |
||||||
|
refs/tags/*,delete) |
||||||
|
# delete tag |
||||||
|
if [ "$allowdeletetag" != "true" ]; then |
||||||
|
echo "*** Deleting a tag is not allowed in this repository" >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
;; |
||||||
|
refs/tags/*,tag) |
||||||
|
# annotated tag |
||||||
|
if [ "$allowmodifytag" != "true" ] && git rev-parse $refname > /dev/null 2>&1 |
||||||
|
then |
||||||
|
echo "*** Tag '$refname' already exists." >&2 |
||||||
|
echo "*** Modifying a tag is not allowed in this repository." >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
;; |
||||||
|
refs/heads/*,commit) |
||||||
|
# branch |
||||||
|
if [ "$oldrev" = "$zero" -a "$denycreatebranch" = "true" ]; then |
||||||
|
echo "*** Creating a branch is not allowed in this repository" >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
;; |
||||||
|
refs/heads/*,delete) |
||||||
|
# delete branch |
||||||
|
if [ "$allowdeletebranch" != "true" ]; then |
||||||
|
echo "*** Deleting a branch is not allowed in this repository" >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
;; |
||||||
|
refs/remotes/*,commit) |
||||||
|
# tracking branch |
||||||
|
;; |
||||||
|
refs/remotes/*,delete) |
||||||
|
# delete tracking branch |
||||||
|
if [ "$allowdeletebranch" != "true" ]; then |
||||||
|
echo "*** Deleting a tracking branch is not allowed in this repository" >&2 |
||||||
|
exit 1 |
||||||
|
fi |
||||||
|
;; |
||||||
|
*) |
||||||
|
# Anything else (is there anything else?) |
||||||
|
echo "*** Update hook: unknown type of update to ref $refname of type $newrev_type" >&2 |
||||||
|
exit 1 |
||||||
|
;; |
||||||
|
esac |
||||||
|
|
||||||
|
# --- Finished |
||||||
|
exit 0 |
@ -0,0 +1,6 @@ |
|||||||
|
# git ls-files --others --exclude-from=.git/info/exclude |
||||||
|
# Lines that start with '#' are comments. |
||||||
|
# For a project mostly in C, the following would be a good set of |
||||||
|
# exclude patterns (uncomment them if you want to use them): |
||||||
|
# *.[oa] |
||||||
|
# *~ |
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1 @@ |
|||||||
|
c7cd3cd144e6d23c9d6f3d07e52b2c1a956e0338 |
@ -0,0 +1,26 @@ |
|||||||
|
- |
||||||
|
id: 1 |
||||||
|
title: First project |
||||||
|
repo_id: 1 |
||||||
|
is_closed: false |
||||||
|
creator_id: 2 |
||||||
|
board_type: 1 |
||||||
|
type: 2 |
||||||
|
|
||||||
|
- |
||||||
|
id: 2 |
||||||
|
title: second project |
||||||
|
repo_id: 3 |
||||||
|
is_closed: false |
||||||
|
creator_id: 3 |
||||||
|
board_type: 1 |
||||||
|
type: 2 |
||||||
|
|
||||||
|
- |
||||||
|
id: 3 |
||||||
|
title: project on repo with disabled project |
||||||
|
repo_id: 4 |
||||||
|
is_closed: true |
||||||
|
creator_id: 5 |
||||||
|
board_type: 1 |
||||||
|
type: 2 |
@ -0,0 +1,23 @@ |
|||||||
|
- |
||||||
|
id: 1 |
||||||
|
project_id: 1 |
||||||
|
title: To Do |
||||||
|
creator_id: 2 |
||||||
|
created_unix: 1588117528 |
||||||
|
updated_unix: 1588117528 |
||||||
|
|
||||||
|
- |
||||||
|
id: 2 |
||||||
|
project_id: 1 |
||||||
|
title: In Progress |
||||||
|
creator_id: 2 |
||||||
|
created_unix: 1588117528 |
||||||
|
updated_unix: 1588117528 |
||||||
|
|
||||||
|
- |
||||||
|
id: 3 |
||||||
|
project_id: 1 |
||||||
|
title: Done |
||||||
|
creator_id: 2 |
||||||
|
created_unix: 1588117528 |
||||||
|
updated_unix: 1588117528 |
@ -0,0 +1,23 @@ |
|||||||
|
- |
||||||
|
id: 1 |
||||||
|
issue_id: 1 |
||||||
|
project_id: 1 |
||||||
|
project_board_id: 1 |
||||||
|
|
||||||
|
- |
||||||
|
id: 2 |
||||||
|
issue_id: 2 |
||||||
|
project_id: 1 |
||||||
|
project_board_id: 0 # no board assigned |
||||||
|
|
||||||
|
- |
||||||
|
id: 3 |
||||||
|
issue_id: 3 |
||||||
|
project_id: 1 |
||||||
|
project_board_id: 2 |
||||||
|
|
||||||
|
- |
||||||
|
id: 4 |
||||||
|
issue_id: 5 |
||||||
|
project_id: 1 |
||||||
|
project_board_id: 3 |
@ -0,0 +1,85 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package migrations |
||||||
|
|
||||||
|
import ( |
||||||
|
"code.gitea.io/gitea/modules/timeutil" |
||||||
|
|
||||||
|
"xorm.io/xorm" |
||||||
|
) |
||||||
|
|
||||||
|
func addProjectsInfo(x *xorm.Engine) error { |
||||||
|
|
||||||
|
// Create new tables
|
||||||
|
type ( |
||||||
|
ProjectType uint8 |
||||||
|
ProjectBoardType uint8 |
||||||
|
) |
||||||
|
|
||||||
|
type Project struct { |
||||||
|
ID int64 `xorm:"pk autoincr"` |
||||||
|
Title string `xorm:"INDEX NOT NULL"` |
||||||
|
Description string `xorm:"TEXT"` |
||||||
|
RepoID int64 `xorm:"INDEX"` |
||||||
|
CreatorID int64 `xorm:"NOT NULL"` |
||||||
|
IsClosed bool `xorm:"INDEX"` |
||||||
|
|
||||||
|
BoardType ProjectBoardType |
||||||
|
Type ProjectType |
||||||
|
|
||||||
|
ClosedDateUnix timeutil.TimeStamp |
||||||
|
CreatedUnix timeutil.TimeStamp `xorm:"INDEX created"` |
||||||
|
UpdatedUnix timeutil.TimeStamp `xorm:"INDEX updated"` |
||||||
|
} |
||||||
|
|
||||||
|
if err := x.Sync2(new(Project)); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
type Comment struct { |
||||||
|
OldProjectID int64 |
||||||
|
ProjectID int64 |
||||||
|
} |
||||||
|
|
||||||
|
if err := x.Sync2(new(Comment)); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
type Repository struct { |
||||||
|
ID int64 |
||||||
|
NumProjects int `xorm:"NOT NULL DEFAULT 0"` |
||||||
|
NumClosedProjects int `xorm:"NOT NULL DEFAULT 0"` |
||||||
|
} |
||||||
|
|
||||||
|
if err := x.Sync2(new(Repository)); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// ProjectIssue saves relation from issue to a project
|
||||||
|
type ProjectIssue struct { |
||||||
|
ID int64 `xorm:"pk autoincr"` |
||||||
|
IssueID int64 `xorm:"INDEX"` |
||||||
|
ProjectID int64 `xorm:"INDEX"` |
||||||
|
ProjectBoardID int64 `xorm:"INDEX"` |
||||||
|
} |
||||||
|
|
||||||
|
if err := x.Sync2(new(ProjectIssue)); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
type ProjectBoard struct { |
||||||
|
ID int64 `xorm:"pk autoincr"` |
||||||
|
Title string |
||||||
|
Default bool `xorm:"NOT NULL DEFAULT false"` |
||||||
|
|
||||||
|
ProjectID int64 `xorm:"INDEX NOT NULL"` |
||||||
|
CreatorID int64 `xorm:"NOT NULL"` |
||||||
|
|
||||||
|
CreatedUnix timeutil.TimeStamp `xorm:"INDEX created"` |
||||||
|
UpdatedUnix timeutil.TimeStamp `xorm:"INDEX updated"` |
||||||
|
} |
||||||
|
|
||||||
|
return x.Sync2(new(ProjectBoard)) |
||||||
|
} |
@ -0,0 +1,307 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package models |
||||||
|
|
||||||
|
import ( |
||||||
|
"errors" |
||||||
|
"fmt" |
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/setting" |
||||||
|
"code.gitea.io/gitea/modules/timeutil" |
||||||
|
"code.gitea.io/gitea/modules/util" |
||||||
|
|
||||||
|
"xorm.io/builder" |
||||||
|
) |
||||||
|
|
||||||
|
type ( |
||||||
|
// ProjectsConfig is used to identify the type of board that is being created
|
||||||
|
ProjectsConfig struct { |
||||||
|
BoardType ProjectBoardType |
||||||
|
Translation string |
||||||
|
} |
||||||
|
|
||||||
|
// ProjectType is used to identify the type of project in question and ownership
|
||||||
|
ProjectType uint8 |
||||||
|
) |
||||||
|
|
||||||
|
const ( |
||||||
|
// ProjectTypeIndividual is a type of project board that is owned by an individual
|
||||||
|
ProjectTypeIndividual ProjectType = iota + 1 |
||||||
|
|
||||||
|
// ProjectTypeRepository is a project that is tied to a repository
|
||||||
|
ProjectTypeRepository |
||||||
|
|
||||||
|
// ProjectTypeOrganization is a project that is tied to an organisation
|
||||||
|
ProjectTypeOrganization |
||||||
|
) |
||||||
|
|
||||||
|
// Project represents a project board
|
||||||
|
type Project struct { |
||||||
|
ID int64 `xorm:"pk autoincr"` |
||||||
|
Title string `xorm:"INDEX NOT NULL"` |
||||||
|
Description string `xorm:"TEXT"` |
||||||
|
RepoID int64 `xorm:"INDEX"` |
||||||
|
CreatorID int64 `xorm:"NOT NULL"` |
||||||
|
IsClosed bool `xorm:"INDEX"` |
||||||
|
BoardType ProjectBoardType |
||||||
|
Type ProjectType |
||||||
|
|
||||||
|
RenderedContent string `xorm:"-"` |
||||||
|
|
||||||
|
CreatedUnix timeutil.TimeStamp `xorm:"INDEX created"` |
||||||
|
UpdatedUnix timeutil.TimeStamp `xorm:"INDEX updated"` |
||||||
|
ClosedDateUnix timeutil.TimeStamp |
||||||
|
} |
||||||
|
|
||||||
|
// GetProjectsConfig retrieves the types of configurations projects could have
|
||||||
|
func GetProjectsConfig() []ProjectsConfig { |
||||||
|
return []ProjectsConfig{ |
||||||
|
{ProjectBoardTypeNone, "repo.projects.type.none"}, |
||||||
|
{ProjectBoardTypeBasicKanban, "repo.projects.type.basic_kanban"}, |
||||||
|
{ProjectBoardTypeBugTriage, "repo.projects.type.bug_triage"}, |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// IsProjectTypeValid checks if a project type is valid
|
||||||
|
func IsProjectTypeValid(p ProjectType) bool { |
||||||
|
switch p { |
||||||
|
case ProjectTypeRepository: |
||||||
|
return true |
||||||
|
default: |
||||||
|
return false |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// ProjectSearchOptions are options for GetProjects
|
||||||
|
type ProjectSearchOptions struct { |
||||||
|
RepoID int64 |
||||||
|
Page int |
||||||
|
IsClosed util.OptionalBool |
||||||
|
SortType string |
||||||
|
Type ProjectType |
||||||
|
} |
||||||
|
|
||||||
|
// GetProjects returns a list of all projects that have been created in the repository
|
||||||
|
func GetProjects(opts ProjectSearchOptions) ([]*Project, int64, error) { |
||||||
|
return getProjects(x, opts) |
||||||
|
} |
||||||
|
|
||||||
|
func getProjects(e Engine, opts ProjectSearchOptions) ([]*Project, int64, error) { |
||||||
|
|
||||||
|
projects := make([]*Project, 0, setting.UI.IssuePagingNum) |
||||||
|
|
||||||
|
var cond builder.Cond = builder.Eq{"repo_id": opts.RepoID} |
||||||
|
switch opts.IsClosed { |
||||||
|
case util.OptionalBoolTrue: |
||||||
|
cond = cond.And(builder.Eq{"is_closed": true}) |
||||||
|
case util.OptionalBoolFalse: |
||||||
|
cond = cond.And(builder.Eq{"is_closed": false}) |
||||||
|
} |
||||||
|
|
||||||
|
if opts.Type > 0 { |
||||||
|
cond = cond.And(builder.Eq{"type": opts.Type}) |
||||||
|
} |
||||||
|
|
||||||
|
count, err := e.Where(cond).Count(new(Project)) |
||||||
|
if err != nil { |
||||||
|
return nil, 0, fmt.Errorf("Count: %v", err) |
||||||
|
} |
||||||
|
|
||||||
|
e = e.Where(cond) |
||||||
|
|
||||||
|
if opts.Page > 0 { |
||||||
|
e = e.Limit(setting.UI.IssuePagingNum, (opts.Page-1)*setting.UI.IssuePagingNum) |
||||||
|
} |
||||||
|
|
||||||
|
switch opts.SortType { |
||||||
|
case "oldest": |
||||||
|
e.Desc("created_unix") |
||||||
|
case "recentupdate": |
||||||
|
e.Desc("updated_unix") |
||||||
|
case "leastupdate": |
||||||
|
e.Asc("updated_unix") |
||||||
|
default: |
||||||
|
e.Asc("created_unix") |
||||||
|
} |
||||||
|
|
||||||
|
return projects, count, e.Find(&projects) |
||||||
|
} |
||||||
|
|
||||||
|
// NewProject creates a new Project
|
||||||
|
func NewProject(p *Project) error { |
||||||
|
if !IsProjectBoardTypeValid(p.BoardType) { |
||||||
|
p.BoardType = ProjectBoardTypeNone |
||||||
|
} |
||||||
|
|
||||||
|
if !IsProjectTypeValid(p.Type) { |
||||||
|
return errors.New("project type is not valid") |
||||||
|
} |
||||||
|
|
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
|
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if _, err := sess.Insert(p); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if _, err := sess.Exec("UPDATE `repository` SET num_projects = num_projects + 1 WHERE id = ?", p.RepoID); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := createBoardsForProjectsType(sess, p); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
// GetProjectByID returns the projects in a repository
|
||||||
|
func GetProjectByID(id int64) (*Project, error) { |
||||||
|
return getProjectByID(x, id) |
||||||
|
} |
||||||
|
|
||||||
|
func getProjectByID(e Engine, id int64) (*Project, error) { |
||||||
|
p := new(Project) |
||||||
|
|
||||||
|
has, err := e.ID(id).Get(p) |
||||||
|
if err != nil { |
||||||
|
return nil, err |
||||||
|
} else if !has { |
||||||
|
return nil, ErrProjectNotExist{ID: id} |
||||||
|
} |
||||||
|
|
||||||
|
return p, nil |
||||||
|
} |
||||||
|
|
||||||
|
// UpdateProject updates project properties
|
||||||
|
func UpdateProject(p *Project) error { |
||||||
|
return updateProject(x, p) |
||||||
|
} |
||||||
|
|
||||||
|
func updateProject(e Engine, p *Project) error { |
||||||
|
_, err := e.ID(p.ID).Cols( |
||||||
|
"title", |
||||||
|
"description", |
||||||
|
).Update(p) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
func updateRepositoryProjectCount(e Engine, repoID int64) error { |
||||||
|
if _, err := e.Exec(builder.Update( |
||||||
|
builder.Eq{ |
||||||
|
"`num_projects`": builder.Select("count(*)").From("`project`"). |
||||||
|
Where(builder.Eq{"`project`.`repo_id`": repoID}. |
||||||
|
And(builder.Eq{"`project`.`type`": ProjectTypeRepository})), |
||||||
|
}).From("`repository`").Where(builder.Eq{"id": repoID})); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if _, err := e.Exec(builder.Update( |
||||||
|
builder.Eq{ |
||||||
|
"`num_closed_projects`": builder.Select("count(*)").From("`project`"). |
||||||
|
Where(builder.Eq{"`project`.`repo_id`": repoID}. |
||||||
|
And(builder.Eq{"`project`.`type`": ProjectTypeRepository}). |
||||||
|
And(builder.Eq{"`project`.`is_closed`": true})), |
||||||
|
}).From("`repository`").Where(builder.Eq{"id": repoID})); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
// ChangeProjectStatusByRepoIDAndID toggles a project between opened and closed
|
||||||
|
func ChangeProjectStatusByRepoIDAndID(repoID, projectID int64, isClosed bool) error { |
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
p := new(Project) |
||||||
|
|
||||||
|
has, err := sess.ID(projectID).Where("repo_id = ?", repoID).Get(p) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} else if !has { |
||||||
|
return ErrProjectNotExist{ID: projectID, RepoID: repoID} |
||||||
|
} |
||||||
|
|
||||||
|
if err := changeProjectStatus(sess, p, isClosed); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
// ChangeProjectStatus toggle a project between opened and closed
|
||||||
|
func ChangeProjectStatus(p *Project, isClosed bool) error { |
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := changeProjectStatus(sess, p, isClosed); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
func changeProjectStatus(e Engine, p *Project, isClosed bool) error { |
||||||
|
p.IsClosed = isClosed |
||||||
|
p.ClosedDateUnix = timeutil.TimeStampNow() |
||||||
|
count, err := e.ID(p.ID).Where("repo_id = ? AND is_closed = ?", p.RepoID, !isClosed).Cols("is_closed", "closed_date_unix").Update(p) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
if count < 1 { |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
return updateRepositoryProjectCount(e, p.RepoID) |
||||||
|
} |
||||||
|
|
||||||
|
// DeleteProjectByID deletes a project from a repository.
|
||||||
|
func DeleteProjectByID(id int64) error { |
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := deleteProjectByID(sess, id); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
func deleteProjectByID(e Engine, id int64) error { |
||||||
|
p, err := getProjectByID(e, id) |
||||||
|
if err != nil { |
||||||
|
if IsErrProjectNotExist(err) { |
||||||
|
return nil |
||||||
|
} |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := deleteProjectIssuesByProjectID(e, id); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := deleteProjectBoardByProjectID(e, id); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if _, err = e.ID(p.ID).Delete(new(Project)); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return updateRepositoryProjectCount(e, p.RepoID) |
||||||
|
} |
@ -0,0 +1,220 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package models |
||||||
|
|
||||||
|
import ( |
||||||
|
"code.gitea.io/gitea/modules/setting" |
||||||
|
"code.gitea.io/gitea/modules/timeutil" |
||||||
|
|
||||||
|
"xorm.io/xorm" |
||||||
|
) |
||||||
|
|
||||||
|
type ( |
||||||
|
// ProjectBoardType is used to represent a project board type
|
||||||
|
ProjectBoardType uint8 |
||||||
|
|
||||||
|
// ProjectBoardList is a list of all project boards in a repository
|
||||||
|
ProjectBoardList []*ProjectBoard |
||||||
|
) |
||||||
|
|
||||||
|
const ( |
||||||
|
// ProjectBoardTypeNone is a project board type that has no predefined columns
|
||||||
|
ProjectBoardTypeNone ProjectBoardType = iota |
||||||
|
|
||||||
|
// ProjectBoardTypeBasicKanban is a project board type that has basic predefined columns
|
||||||
|
ProjectBoardTypeBasicKanban |
||||||
|
|
||||||
|
// ProjectBoardTypeBugTriage is a project board type that has predefined columns suited to hunting down bugs
|
||||||
|
ProjectBoardTypeBugTriage |
||||||
|
) |
||||||
|
|
||||||
|
// ProjectBoard is used to represent boards on a project
|
||||||
|
type ProjectBoard struct { |
||||||
|
ID int64 `xorm:"pk autoincr"` |
||||||
|
Title string |
||||||
|
Default bool `xorm:"NOT NULL DEFAULT false"` // issues not assigned to a specific board will be assigned to this board
|
||||||
|
|
||||||
|
ProjectID int64 `xorm:"INDEX NOT NULL"` |
||||||
|
CreatorID int64 `xorm:"NOT NULL"` |
||||||
|
|
||||||
|
CreatedUnix timeutil.TimeStamp `xorm:"INDEX created"` |
||||||
|
UpdatedUnix timeutil.TimeStamp `xorm:"INDEX updated"` |
||||||
|
|
||||||
|
Issues []*Issue `xorm:"-"` |
||||||
|
} |
||||||
|
|
||||||
|
// IsProjectBoardTypeValid checks if the project board type is valid
|
||||||
|
func IsProjectBoardTypeValid(p ProjectBoardType) bool { |
||||||
|
switch p { |
||||||
|
case ProjectBoardTypeNone, ProjectBoardTypeBasicKanban, ProjectBoardTypeBugTriage: |
||||||
|
return true |
||||||
|
default: |
||||||
|
return false |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func createBoardsForProjectsType(sess *xorm.Session, project *Project) error { |
||||||
|
|
||||||
|
var items []string |
||||||
|
|
||||||
|
switch project.BoardType { |
||||||
|
|
||||||
|
case ProjectBoardTypeBugTriage: |
||||||
|
items = setting.Project.ProjectBoardBugTriageType |
||||||
|
|
||||||
|
case ProjectBoardTypeBasicKanban: |
||||||
|
items = setting.Project.ProjectBoardBasicKanbanType |
||||||
|
|
||||||
|
case ProjectBoardTypeNone: |
||||||
|
fallthrough |
||||||
|
default: |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
if len(items) == 0 { |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
var boards = make([]ProjectBoard, 0, len(items)) |
||||||
|
|
||||||
|
for _, v := range items { |
||||||
|
boards = append(boards, ProjectBoard{ |
||||||
|
CreatedUnix: timeutil.TimeStampNow(), |
||||||
|
CreatorID: project.CreatorID, |
||||||
|
Title: v, |
||||||
|
ProjectID: project.ID, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
_, err := sess.Insert(boards) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// NewProjectBoard adds a new project board to a given project
|
||||||
|
func NewProjectBoard(board *ProjectBoard) error { |
||||||
|
_, err := x.Insert(board) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// DeleteProjectBoardByID removes all issues references to the project board.
|
||||||
|
func DeleteProjectBoardByID(boardID int64) error { |
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := deleteProjectBoardByID(sess, boardID); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
func deleteProjectBoardByID(e Engine, boardID int64) error { |
||||||
|
board, err := getProjectBoard(e, boardID) |
||||||
|
if err != nil { |
||||||
|
if IsErrProjectBoardNotExist(err) { |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err = board.removeIssues(e); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if _, err := e.ID(board.ID).Delete(board); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
func deleteProjectBoardByProjectID(e Engine, projectID int64) error { |
||||||
|
_, err := e.Where("project_id=?", projectID).Delete(&ProjectBoard{}) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// GetProjectBoard fetches the current board of a project
|
||||||
|
func GetProjectBoard(boardID int64) (*ProjectBoard, error) { |
||||||
|
return getProjectBoard(x, boardID) |
||||||
|
} |
||||||
|
|
||||||
|
func getProjectBoard(e Engine, boardID int64) (*ProjectBoard, error) { |
||||||
|
board := new(ProjectBoard) |
||||||
|
|
||||||
|
has, err := e.ID(boardID).Get(board) |
||||||
|
if err != nil { |
||||||
|
return nil, err |
||||||
|
} else if !has { |
||||||
|
return nil, ErrProjectBoardNotExist{BoardID: boardID} |
||||||
|
} |
||||||
|
|
||||||
|
return board, nil |
||||||
|
} |
||||||
|
|
||||||
|
// UpdateProjectBoard updates the title of a project board
|
||||||
|
func UpdateProjectBoard(board *ProjectBoard) error { |
||||||
|
return updateProjectBoard(x, board) |
||||||
|
} |
||||||
|
|
||||||
|
func updateProjectBoard(e Engine, board *ProjectBoard) error { |
||||||
|
_, err := e.ID(board.ID).Cols( |
||||||
|
"title", |
||||||
|
"default", |
||||||
|
).Update(board) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// GetProjectBoards fetches all boards related to a project
|
||||||
|
func GetProjectBoards(projectID int64) ([]*ProjectBoard, error) { |
||||||
|
|
||||||
|
var boards = make([]*ProjectBoard, 0, 5) |
||||||
|
|
||||||
|
sess := x.Where("project_id=?", projectID) |
||||||
|
return boards, sess.Find(&boards) |
||||||
|
} |
||||||
|
|
||||||
|
// GetUncategorizedBoard represents a board for issues not assigned to one
|
||||||
|
func GetUncategorizedBoard(projectID int64) (*ProjectBoard, error) { |
||||||
|
return &ProjectBoard{ |
||||||
|
ProjectID: projectID, |
||||||
|
Title: "Uncategorized", |
||||||
|
Default: true, |
||||||
|
}, nil |
||||||
|
} |
||||||
|
|
||||||
|
// LoadIssues load issues assigned to this board
|
||||||
|
func (b *ProjectBoard) LoadIssues() (IssueList, error) { |
||||||
|
var boardID int64 |
||||||
|
if !b.Default { |
||||||
|
boardID = b.ID |
||||||
|
|
||||||
|
} else { |
||||||
|
// Issues without ProjectBoardID
|
||||||
|
boardID = -1 |
||||||
|
} |
||||||
|
issues, err := Issues(&IssuesOptions{ |
||||||
|
ProjectBoardID: boardID, |
||||||
|
ProjectID: b.ProjectID, |
||||||
|
}) |
||||||
|
b.Issues = issues |
||||||
|
return issues, err |
||||||
|
} |
||||||
|
|
||||||
|
// LoadIssues load issues assigned to the boards
|
||||||
|
func (bs ProjectBoardList) LoadIssues() (IssueList, error) { |
||||||
|
issues := make(IssueList, 0, len(bs)*10) |
||||||
|
for i := range bs { |
||||||
|
il, err := bs[i].LoadIssues() |
||||||
|
if err != nil { |
||||||
|
return nil, err |
||||||
|
} |
||||||
|
bs[i].Issues = il |
||||||
|
issues = append(issues, il...) |
||||||
|
} |
||||||
|
return issues, nil |
||||||
|
} |
@ -0,0 +1,210 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package models |
||||||
|
|
||||||
|
import ( |
||||||
|
"fmt" |
||||||
|
|
||||||
|
"xorm.io/xorm" |
||||||
|
) |
||||||
|
|
||||||
|
// ProjectIssue saves relation from issue to a project
|
||||||
|
type ProjectIssue struct { |
||||||
|
ID int64 `xorm:"pk autoincr"` |
||||||
|
IssueID int64 `xorm:"INDEX"` |
||||||
|
ProjectID int64 `xorm:"INDEX"` |
||||||
|
|
||||||
|
// If 0, then it has not been added to a specific board in the project
|
||||||
|
ProjectBoardID int64 `xorm:"INDEX"` |
||||||
|
} |
||||||
|
|
||||||
|
func deleteProjectIssuesByProjectID(e Engine, projectID int64) error { |
||||||
|
_, err := e.Where("project_id=?", projectID).Delete(&ProjectIssue{}) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// ___
|
||||||
|
// |_ _|___ ___ _ _ ___
|
||||||
|
// | |/ __/ __| | | |/ _ \
|
||||||
|
// | |\__ \__ \ |_| | __/
|
||||||
|
// |___|___/___/\__,_|\___|
|
||||||
|
|
||||||
|
// LoadProject load the project the issue was assigned to
|
||||||
|
func (i *Issue) LoadProject() (err error) { |
||||||
|
return i.loadProject(x) |
||||||
|
} |
||||||
|
|
||||||
|
func (i *Issue) loadProject(e Engine) (err error) { |
||||||
|
if i.Project == nil { |
||||||
|
var p Project |
||||||
|
if _, err = e.Table("project"). |
||||||
|
Join("INNER", "project_issue", "project.id=project_issue.project_id"). |
||||||
|
Where("project_issue.issue_id = ?", i.ID). |
||||||
|
Get(&p); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
i.Project = &p |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
// ProjectID return project id if issue was assigned to one
|
||||||
|
func (i *Issue) ProjectID() int64 { |
||||||
|
return i.projectID(x) |
||||||
|
} |
||||||
|
|
||||||
|
func (i *Issue) projectID(e Engine) int64 { |
||||||
|
var ip ProjectIssue |
||||||
|
has, err := e.Where("issue_id=?", i.ID).Get(&ip) |
||||||
|
if err != nil || !has { |
||||||
|
return 0 |
||||||
|
} |
||||||
|
return ip.ProjectID |
||||||
|
} |
||||||
|
|
||||||
|
// ProjectBoardID return project board id if issue was assigned to one
|
||||||
|
func (i *Issue) ProjectBoardID() int64 { |
||||||
|
return i.projectBoardID(x) |
||||||
|
} |
||||||
|
|
||||||
|
func (i *Issue) projectBoardID(e Engine) int64 { |
||||||
|
var ip ProjectIssue |
||||||
|
has, err := e.Where("issue_id=?", i.ID).Get(&ip) |
||||||
|
if err != nil || !has { |
||||||
|
return 0 |
||||||
|
} |
||||||
|
return ip.ProjectBoardID |
||||||
|
} |
||||||
|
|
||||||
|
// ____ _ _
|
||||||
|
// | _ \ _ __ ___ (_) ___ ___| |_
|
||||||
|
// | |_) | '__/ _ \| |/ _ \/ __| __|
|
||||||
|
// | __/| | | (_) | | __/ (__| |_
|
||||||
|
// |_| |_| \___// |\___|\___|\__|
|
||||||
|
// |__/
|
||||||
|
|
||||||
|
// NumIssues return counter of all issues assigned to a project
|
||||||
|
func (p *Project) NumIssues() int { |
||||||
|
c, err := x.Table("project_issue"). |
||||||
|
Where("project_id=?", p.ID). |
||||||
|
GroupBy("issue_id"). |
||||||
|
Cols("issue_id"). |
||||||
|
Count() |
||||||
|
if err != nil { |
||||||
|
return 0 |
||||||
|
} |
||||||
|
return int(c) |
||||||
|
} |
||||||
|
|
||||||
|
// NumClosedIssues return counter of closed issues assigned to a project
|
||||||
|
func (p *Project) NumClosedIssues() int { |
||||||
|
c, err := x.Table("project_issue"). |
||||||
|
Join("INNER", "issue", "project_issue.issue_id=issue.id"). |
||||||
|
Where("project_issue.project_id=? AND issue.is_closed=?", p.ID, true). |
||||||
|
Cols("issue_id"). |
||||||
|
Count() |
||||||
|
if err != nil { |
||||||
|
return 0 |
||||||
|
} |
||||||
|
return int(c) |
||||||
|
} |
||||||
|
|
||||||
|
// NumOpenIssues return counter of open issues assigned to a project
|
||||||
|
func (p *Project) NumOpenIssues() int { |
||||||
|
c, err := x.Table("project_issue"). |
||||||
|
Join("INNER", "issue", "project_issue.issue_id=issue.id"). |
||||||
|
Where("project_issue.project_id=? AND issue.is_closed=?", p.ID, false).Count("issue.id") |
||||||
|
if err != nil { |
||||||
|
return 0 |
||||||
|
} |
||||||
|
return int(c) |
||||||
|
} |
||||||
|
|
||||||
|
// ChangeProjectAssign changes the project associated with an issue
|
||||||
|
func ChangeProjectAssign(issue *Issue, doer *User, newProjectID int64) error { |
||||||
|
|
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := addUpdateIssueProject(sess, issue, doer, newProjectID); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
func addUpdateIssueProject(e *xorm.Session, issue *Issue, doer *User, newProjectID int64) error { |
||||||
|
|
||||||
|
oldProjectID := issue.projectID(e) |
||||||
|
|
||||||
|
if _, err := e.Where("project_issue.issue_id=?", issue.ID).Delete(&ProjectIssue{}); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if err := issue.loadRepo(e); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if oldProjectID > 0 || newProjectID > 0 { |
||||||
|
if _, err := createComment(e, &CreateCommentOptions{ |
||||||
|
Type: CommentTypeProject, |
||||||
|
Doer: doer, |
||||||
|
Repo: issue.Repo, |
||||||
|
Issue: issue, |
||||||
|
OldProjectID: oldProjectID, |
||||||
|
ProjectID: newProjectID, |
||||||
|
}); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
_, err := e.Insert(&ProjectIssue{ |
||||||
|
IssueID: issue.ID, |
||||||
|
ProjectID: newProjectID, |
||||||
|
}) |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
// ____ _ _ ____ _
|
||||||
|
// | _ \ _ __ ___ (_) ___ ___| |_| __ ) ___ __ _ _ __ __| |
|
||||||
|
// | |_) | '__/ _ \| |/ _ \/ __| __| _ \ / _ \ / _` | '__/ _` |
|
||||||
|
// | __/| | | (_) | | __/ (__| |_| |_) | (_) | (_| | | | (_| |
|
||||||
|
// |_| |_| \___// |\___|\___|\__|____/ \___/ \__,_|_| \__,_|
|
||||||
|
// |__/
|
||||||
|
|
||||||
|
// MoveIssueAcrossProjectBoards move a card from one board to another
|
||||||
|
func MoveIssueAcrossProjectBoards(issue *Issue, board *ProjectBoard) error { |
||||||
|
|
||||||
|
sess := x.NewSession() |
||||||
|
defer sess.Close() |
||||||
|
if err := sess.Begin(); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
var pis ProjectIssue |
||||||
|
has, err := sess.Where("issue_id=?", issue.ID).Get(&pis) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
if !has { |
||||||
|
return fmt.Errorf("issue has to be added to a project first") |
||||||
|
} |
||||||
|
|
||||||
|
pis.ProjectBoardID = board.ID |
||||||
|
if _, err := sess.ID(pis.ID).Cols("project_board_id").Update(&pis); err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
return sess.Commit() |
||||||
|
} |
||||||
|
|
||||||
|
func (pb *ProjectBoard) removeIssues(e Engine) error { |
||||||
|
_, err := e.Exec("UPDATE `project_issue` SET project_board_id = 0 WHERE project_board_id = ? ", pb.ID) |
||||||
|
return err |
||||||
|
} |
@ -0,0 +1,82 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package models |
||||||
|
|
||||||
|
import ( |
||||||
|
"testing" |
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/timeutil" |
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert" |
||||||
|
) |
||||||
|
|
||||||
|
func TestIsProjectTypeValid(t *testing.T) { |
||||||
|
const UnknownType ProjectType = 15 |
||||||
|
|
||||||
|
var cases = []struct { |
||||||
|
typ ProjectType |
||||||
|
valid bool |
||||||
|
}{ |
||||||
|
{ProjectTypeIndividual, false}, |
||||||
|
{ProjectTypeRepository, true}, |
||||||
|
{ProjectTypeOrganization, false}, |
||||||
|
{UnknownType, false}, |
||||||
|
} |
||||||
|
|
||||||
|
for _, v := range cases { |
||||||
|
assert.Equal(t, v.valid, IsProjectTypeValid(v.typ)) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func TestGetProjects(t *testing.T) { |
||||||
|
assert.NoError(t, PrepareTestDatabase()) |
||||||
|
|
||||||
|
projects, _, err := GetProjects(ProjectSearchOptions{RepoID: 1}) |
||||||
|
assert.NoError(t, err) |
||||||
|
|
||||||
|
// 1 value for this repo exists in the fixtures
|
||||||
|
assert.Len(t, projects, 1) |
||||||
|
|
||||||
|
projects, _, err = GetProjects(ProjectSearchOptions{RepoID: 3}) |
||||||
|
assert.NoError(t, err) |
||||||
|
|
||||||
|
// 1 value for this repo exists in the fixtures
|
||||||
|
assert.Len(t, projects, 1) |
||||||
|
} |
||||||
|
|
||||||
|
func TestProject(t *testing.T) { |
||||||
|
assert.NoError(t, PrepareTestDatabase()) |
||||||
|
|
||||||
|
project := &Project{ |
||||||
|
Type: ProjectTypeRepository, |
||||||
|
BoardType: ProjectBoardTypeBasicKanban, |
||||||
|
Title: "New Project", |
||||||
|
RepoID: 1, |
||||||
|
CreatedUnix: timeutil.TimeStampNow(), |
||||||
|
CreatorID: 2, |
||||||
|
} |
||||||
|
|
||||||
|
assert.NoError(t, NewProject(project)) |
||||||
|
|
||||||
|
_, err := GetProjectByID(project.ID) |
||||||
|
assert.NoError(t, err) |
||||||
|
|
||||||
|
// Update project
|
||||||
|
project.Title = "Updated title" |
||||||
|
assert.NoError(t, UpdateProject(project)) |
||||||
|
|
||||||
|
projectFromDB, err := GetProjectByID(project.ID) |
||||||
|
assert.NoError(t, err) |
||||||
|
|
||||||
|
assert.Equal(t, project.Title, projectFromDB.Title) |
||||||
|
|
||||||
|
assert.NoError(t, ChangeProjectStatus(project, true)) |
||||||
|
|
||||||
|
// Retrieve from DB afresh to check if it is truly closed
|
||||||
|
projectFromDB, err = GetProjectByID(project.ID) |
||||||
|
assert.NoError(t, err) |
||||||
|
|
||||||
|
assert.True(t, projectFromDB.IsClosed) |
||||||
|
} |
@ -0,0 +1,24 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package setting |
||||||
|
|
||||||
|
import "code.gitea.io/gitea/modules/log" |
||||||
|
|
||||||
|
// Project settings
|
||||||
|
var ( |
||||||
|
Project = struct { |
||||||
|
ProjectBoardBasicKanbanType []string |
||||||
|
ProjectBoardBugTriageType []string |
||||||
|
}{ |
||||||
|
ProjectBoardBasicKanbanType: []string{"To Do", "In Progress", "Done"}, |
||||||
|
ProjectBoardBugTriageType: []string{"Needs Triage", "High Priority", "Low Priority", "Closed"}, |
||||||
|
} |
||||||
|
) |
||||||
|
|
||||||
|
func newProject() { |
||||||
|
if err := Cfg.Section("project").MapTo(&Project); err != nil { |
||||||
|
log.Fatal("Failed to map Project settings: %v", err) |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,591 @@ |
|||||||
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package repo |
||||||
|
|
||||||
|
import ( |
||||||
|
"fmt" |
||||||
|
"strings" |
||||||
|
|
||||||
|
"code.gitea.io/gitea/models" |
||||||
|
"code.gitea.io/gitea/modules/auth" |
||||||
|
"code.gitea.io/gitea/modules/base" |
||||||
|
"code.gitea.io/gitea/modules/context" |
||||||
|
"code.gitea.io/gitea/modules/markup/markdown" |
||||||
|
"code.gitea.io/gitea/modules/setting" |
||||||
|
"code.gitea.io/gitea/modules/util" |
||||||
|
) |
||||||
|
|
||||||
|
const ( |
||||||
|
tplProjects base.TplName = "repo/projects/list" |
||||||
|
tplProjectsNew base.TplName = "repo/projects/new" |
||||||
|
tplProjectsView base.TplName = "repo/projects/view" |
||||||
|
tplGenericProjectsNew base.TplName = "user/project" |
||||||
|
) |
||||||
|
|
||||||
|
// MustEnableProjects check if projects are enabled in settings
|
||||||
|
func MustEnableProjects(ctx *context.Context) { |
||||||
|
if models.UnitTypeProjects.UnitGlobalDisabled() { |
||||||
|
ctx.NotFound("EnableKanbanBoard", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if ctx.Repo.Repository != nil { |
||||||
|
if !ctx.Repo.CanRead(models.UnitTypeProjects) { |
||||||
|
ctx.NotFound("MustEnableProjects", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// Projects renders the home page of projects
|
||||||
|
func Projects(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.project_board") |
||||||
|
|
||||||
|
sortType := ctx.QueryTrim("sort") |
||||||
|
|
||||||
|
isShowClosed := strings.ToLower(ctx.QueryTrim("state")) == "closed" |
||||||
|
repo := ctx.Repo.Repository |
||||||
|
page := ctx.QueryInt("page") |
||||||
|
if page <= 1 { |
||||||
|
page = 1 |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["OpenCount"] = repo.NumOpenProjects |
||||||
|
ctx.Data["ClosedCount"] = repo.NumClosedProjects |
||||||
|
|
||||||
|
var total int |
||||||
|
if !isShowClosed { |
||||||
|
total = repo.NumOpenProjects |
||||||
|
} else { |
||||||
|
total = repo.NumClosedProjects |
||||||
|
} |
||||||
|
|
||||||
|
projects, count, err := models.GetProjects(models.ProjectSearchOptions{ |
||||||
|
RepoID: repo.ID, |
||||||
|
Page: page, |
||||||
|
IsClosed: util.OptionalBoolOf(isShowClosed), |
||||||
|
SortType: sortType, |
||||||
|
Type: models.ProjectTypeRepository, |
||||||
|
}) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetProjects", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
for i := range projects { |
||||||
|
projects[i].RenderedContent = string(markdown.Render([]byte(projects[i].Description), ctx.Repo.RepoLink, ctx.Repo.Repository.ComposeMetas())) |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["Projects"] = projects |
||||||
|
|
||||||
|
if isShowClosed { |
||||||
|
ctx.Data["State"] = "closed" |
||||||
|
} else { |
||||||
|
ctx.Data["State"] = "open" |
||||||
|
} |
||||||
|
|
||||||
|
numPages := 0 |
||||||
|
if count > 0 { |
||||||
|
numPages = int((int(count) - 1) / setting.UI.IssuePagingNum) |
||||||
|
} |
||||||
|
|
||||||
|
pager := context.NewPagination(total, setting.UI.IssuePagingNum, page, numPages) |
||||||
|
pager.AddParam(ctx, "state", "State") |
||||||
|
ctx.Data["Page"] = pager |
||||||
|
|
||||||
|
ctx.Data["IsShowClosed"] = isShowClosed |
||||||
|
ctx.Data["IsProjectsPage"] = true |
||||||
|
ctx.Data["SortType"] = sortType |
||||||
|
|
||||||
|
ctx.HTML(200, tplProjects) |
||||||
|
} |
||||||
|
|
||||||
|
// NewProject render creating a project page
|
||||||
|
func NewProject(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.new") |
||||||
|
ctx.Data["ProjectTypes"] = models.GetProjectsConfig() |
||||||
|
|
||||||
|
ctx.HTML(200, tplProjectsNew) |
||||||
|
} |
||||||
|
|
||||||
|
// NewRepoProjectPost creates a new project
|
||||||
|
func NewRepoProjectPost(ctx *context.Context, form auth.CreateProjectForm) { |
||||||
|
|
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.new") |
||||||
|
|
||||||
|
if ctx.HasError() { |
||||||
|
ctx.HTML(200, tplProjectsNew) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.NewProject(&models.Project{ |
||||||
|
RepoID: ctx.Repo.Repository.ID, |
||||||
|
Title: form.Title, |
||||||
|
Description: form.Content, |
||||||
|
CreatorID: ctx.User.ID, |
||||||
|
BoardType: form.BoardType, |
||||||
|
Type: models.ProjectTypeRepository, |
||||||
|
}); err != nil { |
||||||
|
ctx.ServerError("NewProject", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.create_success", form.Title)) |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects") |
||||||
|
} |
||||||
|
|
||||||
|
// ChangeProjectStatus updates the status of a project between "open" and "close"
|
||||||
|
func ChangeProjectStatus(ctx *context.Context) { |
||||||
|
toClose := false |
||||||
|
switch ctx.Params(":action") { |
||||||
|
case "open": |
||||||
|
toClose = false |
||||||
|
case "close": |
||||||
|
toClose = true |
||||||
|
default: |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects") |
||||||
|
} |
||||||
|
id := ctx.ParamsInt64(":id") |
||||||
|
|
||||||
|
if err := models.ChangeProjectStatusByRepoIDAndID(ctx.Repo.Repository.ID, id, toClose); err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", err) |
||||||
|
} else { |
||||||
|
ctx.ServerError("ChangeProjectStatusByIDAndRepoID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects?state=" + ctx.Params(":action")) |
||||||
|
} |
||||||
|
|
||||||
|
// DeleteProject delete a project
|
||||||
|
func DeleteProject(ctx *context.Context) { |
||||||
|
p, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.DeleteProjectByID(p.ID); err != nil { |
||||||
|
ctx.Flash.Error("DeleteProjectByID: " + err.Error()) |
||||||
|
} else { |
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.deletion_success")) |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(200, map[string]interface{}{ |
||||||
|
"redirect": ctx.Repo.RepoLink + "/projects", |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// EditProject allows a project to be edited
|
||||||
|
func EditProject(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.edit") |
||||||
|
ctx.Data["PageIsProjects"] = true |
||||||
|
ctx.Data["PageIsEditProjects"] = true |
||||||
|
|
||||||
|
p, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["title"] = p.Title |
||||||
|
ctx.Data["content"] = p.Description |
||||||
|
|
||||||
|
ctx.HTML(200, tplProjectsNew) |
||||||
|
} |
||||||
|
|
||||||
|
// EditProjectPost response for editing a project
|
||||||
|
func EditProjectPost(ctx *context.Context, form auth.CreateProjectForm) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.edit") |
||||||
|
ctx.Data["PageIsProjects"] = true |
||||||
|
ctx.Data["PageIsEditProjects"] = true |
||||||
|
|
||||||
|
if ctx.HasError() { |
||||||
|
ctx.HTML(200, tplMilestoneNew) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
p, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
p.Title = form.Title |
||||||
|
p.Description = form.Content |
||||||
|
if err = models.UpdateProject(p); err != nil { |
||||||
|
ctx.ServerError("UpdateProjects", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.edit_success", p.Title)) |
||||||
|
ctx.Redirect(ctx.Repo.RepoLink + "/projects") |
||||||
|
} |
||||||
|
|
||||||
|
// ViewProject renders the project board for a project
|
||||||
|
func ViewProject(ctx *context.Context) { |
||||||
|
|
||||||
|
project, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if project.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
uncategorizedBoard, err := models.GetUncategorizedBoard(project.ID) |
||||||
|
uncategorizedBoard.Title = ctx.Tr("repo.projects.type.uncategorized") |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetUncategorizedBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
boards, err := models.GetProjectBoards(project.ID) |
||||||
|
if err != nil { |
||||||
|
ctx.ServerError("GetProjectBoards", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
allBoards := models.ProjectBoardList{uncategorizedBoard} |
||||||
|
allBoards = append(allBoards, boards...) |
||||||
|
|
||||||
|
if ctx.Data["Issues"], err = allBoards.LoadIssues(); err != nil { |
||||||
|
ctx.ServerError("LoadIssuesOfBoards", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["Project"] = project |
||||||
|
ctx.Data["Boards"] = allBoards |
||||||
|
ctx.Data["PageIsProjects"] = true |
||||||
|
ctx.Data["RequiresDraggable"] = true |
||||||
|
|
||||||
|
ctx.HTML(200, tplProjectsView) |
||||||
|
} |
||||||
|
|
||||||
|
// UpdateIssueProject change an issue's project
|
||||||
|
func UpdateIssueProject(ctx *context.Context) { |
||||||
|
issues := getActionIssues(ctx) |
||||||
|
if ctx.Written() { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
projectID := ctx.QueryInt64("id") |
||||||
|
for _, issue := range issues { |
||||||
|
oldProjectID := issue.ProjectID() |
||||||
|
if oldProjectID == projectID { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.ChangeProjectAssign(issue, ctx.User, projectID); err != nil { |
||||||
|
ctx.ServerError("ChangeProjectAssign", err) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(200, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// DeleteProjectBoard allows for the deletion of a project board
|
||||||
|
func DeleteProjectBoard(ctx *context.Context) { |
||||||
|
if ctx.User == nil { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only signed in users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if !ctx.Repo.IsOwner() && !ctx.Repo.IsAdmin() && !ctx.Repo.CanAccess(models.AccessModeWrite, models.UnitTypeProjects) { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only authorized users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
project, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
pb, err := models.GetProjectBoard(ctx.ParamsInt64(":boardID")) |
||||||
|
if err != nil { |
||||||
|
ctx.InternalServerError(err) |
||||||
|
return |
||||||
|
} |
||||||
|
if pb.ProjectID != ctx.ParamsInt64(":id") { |
||||||
|
ctx.JSON(422, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Project[%d] as expected", pb.ID, project.ID), |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if project.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.JSON(422, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Repository[%d] as expected", pb.ID, ctx.Repo.Repository.ID), |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.DeleteProjectBoardByID(ctx.ParamsInt64(":boardID")); err != nil { |
||||||
|
ctx.ServerError("DeleteProjectBoardByID", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(200, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// AddBoardToProjectPost allows a new board to be added to a project.
|
||||||
|
func AddBoardToProjectPost(ctx *context.Context, form auth.EditProjectBoardTitleForm) { |
||||||
|
|
||||||
|
if !ctx.Repo.IsOwner() && !ctx.Repo.IsAdmin() && !ctx.Repo.CanAccess(models.AccessModeWrite, models.UnitTypeProjects) { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only authorized users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
project, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.NewProjectBoard(&models.ProjectBoard{ |
||||||
|
ProjectID: project.ID, |
||||||
|
Title: form.Title, |
||||||
|
CreatorID: ctx.User.ID, |
||||||
|
}); err != nil { |
||||||
|
ctx.ServerError("NewProjectBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(200, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// EditProjectBoardTitle allows a project board's title to be updated
|
||||||
|
func EditProjectBoardTitle(ctx *context.Context, form auth.EditProjectBoardTitleForm) { |
||||||
|
|
||||||
|
if ctx.User == nil { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only signed in users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if !ctx.Repo.IsOwner() && !ctx.Repo.IsAdmin() && !ctx.Repo.CanAccess(models.AccessModeWrite, models.UnitTypeProjects) { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only authorized users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
project, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
board, err := models.GetProjectBoard(ctx.ParamsInt64(":boardID")) |
||||||
|
if err != nil { |
||||||
|
ctx.InternalServerError(err) |
||||||
|
return |
||||||
|
} |
||||||
|
if board.ProjectID != ctx.ParamsInt64(":id") { |
||||||
|
ctx.JSON(422, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Project[%d] as expected", board.ID, project.ID), |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if project.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.JSON(422, map[string]string{ |
||||||
|
"message": fmt.Sprintf("ProjectBoard[%d] is not in Repository[%d] as expected", board.ID, ctx.Repo.Repository.ID), |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if form.Title != "" { |
||||||
|
board.Title = form.Title |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.UpdateProjectBoard(board); err != nil { |
||||||
|
ctx.ServerError("UpdateProjectBoard", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(200, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// MoveIssueAcrossBoards move a card from one board to another in a project
|
||||||
|
func MoveIssueAcrossBoards(ctx *context.Context) { |
||||||
|
|
||||||
|
if ctx.User == nil { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only signed in users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if !ctx.Repo.IsOwner() && !ctx.Repo.IsAdmin() && !ctx.Repo.CanAccess(models.AccessModeWrite, models.UnitTypeProjects) { |
||||||
|
ctx.JSON(403, map[string]string{ |
||||||
|
"message": "Only authorized users are allowed to perform this action.", |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
p, err := models.GetProjectByID(ctx.ParamsInt64(":id")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectByID", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if p.RepoID != ctx.Repo.Repository.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
var board *models.ProjectBoard |
||||||
|
|
||||||
|
if ctx.ParamsInt64(":boardID") == 0 { |
||||||
|
|
||||||
|
board = &models.ProjectBoard{ |
||||||
|
ID: 0, |
||||||
|
ProjectID: 0, |
||||||
|
Title: ctx.Tr("repo.projects.type.uncategorized"), |
||||||
|
} |
||||||
|
|
||||||
|
} else { |
||||||
|
board, err = models.GetProjectBoard(ctx.ParamsInt64(":boardID")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrProjectBoardNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetProjectBoard", err) |
||||||
|
} |
||||||
|
return |
||||||
|
} |
||||||
|
if board.ProjectID != p.ID { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
return |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
issue, err := models.GetIssueByID(ctx.ParamsInt64(":index")) |
||||||
|
if err != nil { |
||||||
|
if models.IsErrIssueNotExist(err) { |
||||||
|
ctx.NotFound("", nil) |
||||||
|
} else { |
||||||
|
ctx.ServerError("GetIssueByID", err) |
||||||
|
} |
||||||
|
|
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.MoveIssueAcrossProjectBoards(issue, board); err != nil { |
||||||
|
ctx.ServerError("MoveIssueAcrossProjectBoards", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.JSON(200, map[string]interface{}{ |
||||||
|
"ok": true, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
// CreateProject renders the generic project creation page
|
||||||
|
func CreateProject(ctx *context.Context) { |
||||||
|
ctx.Data["Title"] = ctx.Tr("repo.projects.new") |
||||||
|
ctx.Data["ProjectTypes"] = models.GetProjectsConfig() |
||||||
|
|
||||||
|
ctx.HTML(200, tplGenericProjectsNew) |
||||||
|
} |
||||||
|
|
||||||
|
// CreateProjectPost creates an individual and/or organization project
|
||||||
|
func CreateProjectPost(ctx *context.Context, form auth.UserCreateProjectForm) { |
||||||
|
|
||||||
|
user := checkContextUser(ctx, form.UID) |
||||||
|
if ctx.Written() { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Data["ContextUser"] = user |
||||||
|
|
||||||
|
if ctx.HasError() { |
||||||
|
ctx.HTML(200, tplGenericProjectsNew) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
var projectType = models.ProjectTypeIndividual |
||||||
|
if user.IsOrganization() { |
||||||
|
projectType = models.ProjectTypeOrganization |
||||||
|
} |
||||||
|
|
||||||
|
if err := models.NewProject(&models.Project{ |
||||||
|
Title: form.Title, |
||||||
|
Description: form.Content, |
||||||
|
CreatorID: user.ID, |
||||||
|
BoardType: form.BoardType, |
||||||
|
Type: projectType, |
||||||
|
}); err != nil { |
||||||
|
ctx.ServerError("NewProject", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
ctx.Flash.Success(ctx.Tr("repo.projects.create_success", form.Title)) |
||||||
|
ctx.Redirect(setting.AppSubURL + "/") |
||||||
|
} |
@ -0,0 +1,99 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="repository milestones"> |
||||||
|
{{template "repo/header" .}} |
||||||
|
<div class="ui container"> |
||||||
|
<div class="navbar"> |
||||||
|
{{template "repo/issue/navbar" .}} |
||||||
|
{{if and (or .CanWriteIssues .CanWritePulls) (not .Repository.IsArchived)}} |
||||||
|
<div class="ui right"> |
||||||
|
<a class="ui green button" href="{{$.Link}}/new">{{.i18n.Tr "repo.projects.new"}}</a> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
{{template "base/alert" .}} |
||||||
|
<div class="ui tiny basic buttons"> |
||||||
|
<a class="ui {{if not .IsShowClosed}}green active{{end}} basic button" href="{{.RepoLink}}/projects?state=open"> |
||||||
|
{{svg "octicon-project" 16}} |
||||||
|
{{.i18n.Tr "repo.issues.open_tab" .OpenCount}} |
||||||
|
</a> |
||||||
|
<a class="ui {{if .IsShowClosed}}red active{{end}} basic button" href="{{.RepoLink}}/projects?state=closed"> |
||||||
|
{{svg "octicon-check" 16}} |
||||||
|
{{.i18n.Tr "repo.milestones.close_tab" .ClosedCount}} |
||||||
|
</a> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="ui right floated secondary filter menu"> |
||||||
|
<!-- Sort --> |
||||||
|
<div class="ui dropdown type jump item"> |
||||||
|
<span class="text"> |
||||||
|
{{.i18n.Tr "repo.issues.filter_sort"}} |
||||||
|
<i class="dropdown icon"></i> |
||||||
|
</span> |
||||||
|
<div class="menu"> |
||||||
|
<a class="{{if eq .SortType "oldest"}}active{{end}} item" href="{{$.Link}}?q={{$.Keyword}}&sort=oldest&state={{$.State}}">{{.i18n.Tr "repo.issues.filter_sort.oldest"}}</a> |
||||||
|
<a class="{{if eq .SortType "recentupdate"}}active{{end}} item" href="{{$.Link}}?q={{$.Keyword}}&sort=recentupdate&state={{$.State}}">{{.i18n.Tr "repo.issues.filter_sort.recentupdate"}}</a> |
||||||
|
<a class="{{if eq .SortType "leastupdate"}}active{{end}} item" href="{{$.Link}}?q={{$.Keyword}}&sort=leastupdate&state={{$.State}}">{{.i18n.Tr "repo.issues.filter_sort.leastupdate"}}</a> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<div class="milestone list"> |
||||||
|
{{range .Projects}} |
||||||
|
<li class="item"> |
||||||
|
{{svg "octicon-project" 16}} <a href="{{$.RepoLink}}/projects/{{.ID}}">{{.Title}}</a> |
||||||
|
<div class="meta"> |
||||||
|
{{ $closedDate:= TimeSinceUnix .ClosedDateUnix $.Lang }} |
||||||
|
{{if .IsClosed }} |
||||||
|
{{svg "octicon-clock" 16}} {{$.i18n.Tr "repo.milestones.closed" $closedDate|Str2html}} |
||||||
|
{{end}} |
||||||
|
<span class="issue-stats"> |
||||||
|
{{svg "octicon-issue-opened" 16}} {{$.i18n.Tr "repo.issues.open_tab" .NumOpenIssues}} |
||||||
|
{{svg "octicon-issue-closed" 16}} {{$.i18n.Tr "repo.issues.close_tab" .NumClosedIssues}} |
||||||
|
</span> |
||||||
|
</div> |
||||||
|
{{if and (or $.CanWriteIssues $.CanWritePulls) (not $.Repository.IsArchived)}} |
||||||
|
<div class="ui right operate"> |
||||||
|
<a href="{{$.Link}}/{{.ID}}/edit" data-id={{.ID}} data-title={{.Title}}>{{svg "octicon-pencil" 16}} {{$.i18n.Tr "repo.issues.label_edit"}}</a> |
||||||
|
{{if .IsClosed}} |
||||||
|
<a class="link-action" href data-url="{{$.Link}}/{{.ID}}/open">{{svg "octicon-check" 16}} {{$.i18n.Tr "repo.projects.open"}}</a> |
||||||
|
{{else}} |
||||||
|
<a class="link-action" href data-url="{{$.Link}}/{{.ID}}/close">{{svg "octicon-x" 16}} {{$.i18n.Tr "repo.projects.close"}}</a> |
||||||
|
{{end}} |
||||||
|
<a class="delete-button" href="#" data-url="{{$.RepoLink}}/projects/{{.ID}}/delete" data-id="{{.ID}}">{{svg "octicon-trashcan" 16}} {{$.i18n.Tr "repo.issues.label_delete"}}</a> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
{{if .Description}} |
||||||
|
<div class="content"> |
||||||
|
{{.RenderedContent|Str2html}} |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</li> |
||||||
|
{{end}} |
||||||
|
|
||||||
|
{{template "base/paginate" .}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
{{if or .CanWriteIssues .CanWritePulls}} |
||||||
|
<div class="ui small basic delete modal"> |
||||||
|
<div class="ui icon header"> |
||||||
|
<i class="trash icon"></i> |
||||||
|
{{.i18n.Tr "repo.projects.deletion"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<p>{{.i18n.Tr "repo.projects.deletion_desc"}}</p> |
||||||
|
</div> |
||||||
|
<div class="actions"> |
||||||
|
<div class="ui red basic inverted cancel button"> |
||||||
|
<i class="remove icon"></i> |
||||||
|
{{.i18n.Tr "modal.no"}} |
||||||
|
</div> |
||||||
|
<div class="ui green basic inverted ok button"> |
||||||
|
<i class="checkmark icon"></i> |
||||||
|
{{.i18n.Tr "modal.yes"}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,70 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="repository new milestone"> |
||||||
|
{{template "repo/header" .}} |
||||||
|
<div class="ui container"> |
||||||
|
<div class="navbar"> |
||||||
|
{{template "repo/issue/navbar" .}} |
||||||
|
{{if and (or .CanWriteIssues .CanWritePulls) .PageIsEditProject}} |
||||||
|
<div class="ui right floated secondary menu"> |
||||||
|
<a class="ui green button" href="{{$.RepoLink}}/projects/new">{{.i18n.Tr "repo.milestones.new"}}</a> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
<h2 class="ui dividing header"> |
||||||
|
{{if .PageIsEditProjects}} |
||||||
|
{{.i18n.Tr "repo.projects.edit"}} |
||||||
|
<div class="sub header">{{.i18n.Tr "repo.projects.edit_subheader"}}</div> |
||||||
|
{{else}} |
||||||
|
{{.i18n.Tr "repo.projects.new"}} |
||||||
|
<div class="sub header">{{.i18n.Tr "repo.projects.new_subheader"}}</div> |
||||||
|
{{end}} |
||||||
|
</h2> |
||||||
|
{{template "base/alert" .}} |
||||||
|
<form class="ui form grid" action="{{.Link}}" method="post"> |
||||||
|
{{.CsrfTokenHtml}} |
||||||
|
<div class="eleven wide column"> |
||||||
|
<div class="field {{if .Err_Title}}error{{end}}"> |
||||||
|
<label>{{.i18n.Tr "repo.projects.title"}}</label> |
||||||
|
<input name="title" placeholder="{{.i18n.Tr "repo.projects.title"}}" value="{{.title}}" autofocus required> |
||||||
|
</div> |
||||||
|
<div class="field"> |
||||||
|
<label>{{.i18n.Tr "repo.projects.desc"}}</label> |
||||||
|
<textarea name="content">{{.content}}</textarea> |
||||||
|
</div> |
||||||
|
|
||||||
|
{{if not .PageIsEditProjects}} |
||||||
|
<label>{{.i18n.Tr "repo.projects.template.desc"}}</label> |
||||||
|
<div class="ui selection dropdown"> |
||||||
|
<input type="hidden" name="board_type" value="{{.type}}"> |
||||||
|
<div class="default text">{{.i18n.Tr "repo.projects.template.desc_helper"}}</div> |
||||||
|
<div class="menu"> |
||||||
|
{{range $element := .ProjectTypes}} |
||||||
|
<div class="item" data-id="{{$element.BoardType}}" data-value="{{$element.BoardType}}">{{$.i18n.Tr $element.Translation}}</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
<div class="ui container"> |
||||||
|
<div class="ui divider"></div> |
||||||
|
<div class="ui left"> |
||||||
|
{{if .PageIsEditProjects}} |
||||||
|
<a class="ui blue basic button" href="{{.RepoLink}}/projects"> |
||||||
|
{{.i18n.Tr "repo.milestones.cancel"}} |
||||||
|
</a> |
||||||
|
<button class="ui green button"> |
||||||
|
{{.i18n.Tr "repo.projects.modify"}} |
||||||
|
</button> |
||||||
|
{{else}} |
||||||
|
<button class="ui green button"> |
||||||
|
{{.i18n.Tr "repo.projects.create"}} |
||||||
|
</button> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,153 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="repository"> |
||||||
|
{{template "repo/header" .}} |
||||||
|
<div class="ui container"> |
||||||
|
<div class="ui three column stackable grid"> |
||||||
|
<div class="column"> |
||||||
|
{{template "repo/issue/navbar" .}} |
||||||
|
</div> |
||||||
|
<div class="column center aligned"> |
||||||
|
{{template "repo/issue/search" .}} |
||||||
|
</div> |
||||||
|
<div class="column right aligned"> |
||||||
|
{{if .PageIsProjects}} |
||||||
|
<a class="ui green button show-modal item" data-modal="#new-board-item">{{.i18n.Tr "new_project_board"}}</a> |
||||||
|
{{end}} |
||||||
|
|
||||||
|
<div class="ui small modal" id="new-board-item"> |
||||||
|
<div class="header"> |
||||||
|
{{$.i18n.Tr "repo.projects.board.new"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<form class="ui form"> |
||||||
|
<div class="required field"> |
||||||
|
<label for="new_board">{{$.i18n.Tr "repo.projects.board.new_title"}}</label> |
||||||
|
<input class="new-board" id="new_board" name="title" required> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.i18n.Tr "settings.cancel"}}</div> |
||||||
|
<button data-url="{{$.RepoLink}}/projects/{{$.Project.ID}}" class="ui green button" id="new_board_submit">{{$.i18n.Tr "repo.projects.board.new_submit"}}</button> |
||||||
|
</div> |
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="ui container fluid padded" id="project-board"> |
||||||
|
|
||||||
|
<div class="board"> |
||||||
|
{{ range $board := .Boards }} |
||||||
|
|
||||||
|
<div class="ui segment board-column"> |
||||||
|
<div class="board-column-header"> |
||||||
|
<div class="ui large label board-label">{{.Title}}</div> |
||||||
|
|
||||||
|
{{ if $.IsSigned }} |
||||||
|
{{ if not (eq .ID 0) }} |
||||||
|
<div class="ui dropdown jump item poping up right" data-variation="tiny inverted"> |
||||||
|
<span class="ui text"> |
||||||
|
<img class="ui tiny avatar image" width="24" height="24"> |
||||||
|
<span class="fitted not-mobile" tabindex="-1">{{svg "octicon-kebab-horizontal" 24}}</span> |
||||||
|
</span> |
||||||
|
<div class="menu user-menu" tabindex="-1"> |
||||||
|
<a class="item show-modal button" data-modal="#edit-project-board-modal-{{.ID}}"> |
||||||
|
{{svg "octicon-pencil" 16}} |
||||||
|
{{$.i18n.Tr "repo.projects.board.edit"}} |
||||||
|
</a> |
||||||
|
<a class="item show-modal button" data-modal="#delete-board-modal-{{.ID}}"> |
||||||
|
{{svg "octicon-trashcan" 16}} |
||||||
|
{{$.i18n.Tr "repo.projects.board.delete"}} |
||||||
|
</a> |
||||||
|
|
||||||
|
<div class="ui small modal edit-project-board" id="edit-project-board-modal-{{.ID}}"> |
||||||
|
<div class="header"> |
||||||
|
{{$.i18n.Tr "repo.projects.board.edit"}} |
||||||
|
</div> |
||||||
|
<div class="content"> |
||||||
|
<form class="ui form"> |
||||||
|
<div class="required field"> |
||||||
|
<label for="new_board_title">{{$.i18n.Tr "repo.projects.board.edit_title"}}</label> |
||||||
|
<input class="project-board-title" id="new_board_title" name="title" value="{{.Title}}" required> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.i18n.Tr "settings.cancel"}}</div> |
||||||
|
<button data-url="{{$.RepoLink}}/projects/{{$.Project.ID}}/{{.ID}}" class="ui red button">{{$.i18n.Tr "repo.projects.board.edit"}}</button> |
||||||
|
</div> |
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="ui basic modal" id="delete-board-modal-{{.ID}}"> |
||||||
|
<div class="ui icon header"> |
||||||
|
{{$.i18n.Tr "repo.projects.board.delete"}} |
||||||
|
</div> |
||||||
|
<div class="content center"> |
||||||
|
<input type="hidden" name="action" value="delete"> |
||||||
|
<div class="field"> |
||||||
|
<label> |
||||||
|
{{$.i18n.Tr "repo.projects.board.deletion_desc"}} |
||||||
|
</label> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<form class="ui form" method="post"> |
||||||
|
<div class="text right actions"> |
||||||
|
<div class="ui cancel button">{{$.i18n.Tr "settings.cancel"}}</div> |
||||||
|
<button class="ui red button delete-project-board" data-url="{{$.RepoLink}}/projects/{{$.Project.ID}}/{{.ID}}">{{$.i18n.Tr "repo.projects.board.delete"}}</button> |
||||||
|
</div> |
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{ end }} |
||||||
|
{{ end }} |
||||||
|
</div> |
||||||
|
<div class="ui divider"></div> |
||||||
|
|
||||||
|
<div class="ui cards board" data-url="{{$.RepoLink}}/projects/{{$.Project.ID}}/{{.ID}}" data-project="{{$.Project.ID}}" data-board="{{.ID}}" id="board_{{.ID}}"> |
||||||
|
|
||||||
|
{{ range .Issues }} |
||||||
|
|
||||||
|
<!-- start issue card --> |
||||||
|
<div class="card board-card" data-issue="{{.ID}}"> |
||||||
|
<div class="content"> |
||||||
|
<div class="header"> |
||||||
|
<span class="{{if .IsClosed}}red{{else}}green{{end}}"> |
||||||
|
{{if .IsPull}}{{svg "octicon-git-merge" 16}} |
||||||
|
{{else if .IsClosed}}{{svg "octicon-issue-closed" 16}} |
||||||
|
{{else}}{{svg "octicon-issue-opened" 16}} |
||||||
|
{{end}} |
||||||
|
</span> |
||||||
|
<a class="project-board-title" href="{{$.RepoLink}}/issues/{{.Index}}">#{{.Index}} {{.Title}}</a> |
||||||
|
</div> |
||||||
|
<div class="meta"> |
||||||
|
{{ if .MilestoneID }} |
||||||
|
<a class="milestone" href="{{$.RepoLink}}/milestone/{{ .MilestoneID}}"> |
||||||
|
{{svg "octicon-milestone" 16}} {{ .Milestone.Name }} |
||||||
|
</a> |
||||||
|
{{ end }} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<div class="extra content"> |
||||||
|
{{ range .Labels }} |
||||||
|
<a class="ui label has-emoji" href="{{$.RepoLink}}/issues?labels={{.ID}}" style="color: {{.ForegroundColor}}; background-color: {{.Color}}; margin-bottom: 3px;" title="{{.Description}}">{{.Name}}</a> |
||||||
|
{{ end }} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
<!-- stop issue card --> |
||||||
|
|
||||||
|
{{ end }} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{ end }} |
||||||
|
</div> |
||||||
|
|
||||||
|
</div> |
||||||
|
|
||||||
|
</div> |
||||||
|
|
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,68 @@ |
|||||||
|
{{template "base/head" .}} |
||||||
|
<div class="repository new repo"> |
||||||
|
<div class="ui middle very relaxed page grid"> |
||||||
|
<div class="column"> |
||||||
|
<form class="ui form" action="{{.Link}}" method="post"> |
||||||
|
{{.CsrfTokenHtml}} |
||||||
|
<h3 class="ui top attached header"> |
||||||
|
{{.i18n.Tr "new_project"}} |
||||||
|
</h3> |
||||||
|
<div class="ui attached segment"> |
||||||
|
{{template "base/alert" .}} |
||||||
|
<div class="inline required field {{if .Err_Owner}}error{{end}}"> |
||||||
|
<label>{{.i18n.Tr "repo.owner"}}</label> |
||||||
|
<div class="ui selection owner dropdown"> |
||||||
|
<input type="hidden" id="uid" name="uid" value="{{.ContextUser.ID}}" required> |
||||||
|
<span class="text" title="{{.ContextUser.Name}}"> |
||||||
|
<img class="ui mini image" src="{{.ContextUser.RelAvatarLink}}"> |
||||||
|
{{.ContextUser.ShortName 20}} |
||||||
|
</span> |
||||||
|
<i class="dropdown icon"></i> |
||||||
|
<div class="menu"> |
||||||
|
<div class="item" data-value="{{.SignedUser.ID}}" title="{{.SignedUser.Name}}"> |
||||||
|
<img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}"> {{.SignedUser.ShortName 20}} |
||||||
|
</div> |
||||||
|
{{range .Orgs}} |
||||||
|
<div class="item" data-value="{{.ID}}" title="{{.Name}}"> |
||||||
|
<img class="ui mini image" src="{{.RelAvatarLink}}"> {{.ShortName 20}} |
||||||
|
</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="inline field {{if .Err_Title}}error{{end}}"> |
||||||
|
<label>{{.i18n.Tr "repo.projects.title"}}</label> |
||||||
|
<input name="title" placeholder="{{.i18n.Tr "repo.projects.title"}}" value="{{.title}}" autofocus required> |
||||||
|
</div> |
||||||
|
<div class="inline field"> |
||||||
|
<label>{{.i18n.Tr "repo.projects.desc"}}</label> |
||||||
|
<textarea name="content">{{.content}}</textarea> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="inline field"> |
||||||
|
<label>{{.i18n.Tr "repo.projects.template.desc"}}</label> |
||||||
|
<div class="ui selection dropdown"> |
||||||
|
<input type="hidden" name="board_type" value="{{.type}}"> |
||||||
|
<div class="default text">{{.i18n.Tr "repo.projects.template.desc_helper"}}</div> |
||||||
|
<div class="menu"> |
||||||
|
{{range $element := .ProjectTypes}} |
||||||
|
<div class="item" data-id="{{$element.BoardType}}" data-value="{{$element.BoardType}}">{{$.i18n.Tr $element.Translation}}</div> |
||||||
|
{{end}} |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
|
||||||
|
<div class="inline field"> |
||||||
|
<label></label> |
||||||
|
<button class="ui green button"> |
||||||
|
{{.i18n.Tr "repo.projects.create" }} |
||||||
|
</button> |
||||||
|
<a class="ui button" href="{{AppSubUrl}}/">{{.i18n.Tr "cancel"}}</a> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</form> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
</div> |
||||||
|
{{template "base/footer" .}} |
@ -0,0 +1,99 @@ |
|||||||
|
const {csrf} = window.config; |
||||||
|
|
||||||
|
export default async function initProject() { |
||||||
|
if (!window.config || !window.config.PageIsProjects) { |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
const {Sortable} = await import(/* webpackChunkName: "sortable" */'sortablejs'); |
||||||
|
const boardColumns = document.getElementsByClassName('board-column'); |
||||||
|
|
||||||
|
for (const column of boardColumns) { |
||||||
|
new Sortable( |
||||||
|
column.getElementsByClassName('board')[0], |
||||||
|
{ |
||||||
|
group: 'shared', |
||||||
|
animation: 150, |
||||||
|
onAdd: (e) => { |
||||||
|
$.ajax(`${e.to.dataset.url}/${e.item.dataset.issue}`, { |
||||||
|
headers: { |
||||||
|
'X-Csrf-Token': csrf, |
||||||
|
'X-Remote': true, |
||||||
|
}, |
||||||
|
contentType: 'application/json', |
||||||
|
type: 'POST', |
||||||
|
error: () => { |
||||||
|
e.from.insertBefore(e.item, e.from.children[e.oldIndex]); |
||||||
|
}, |
||||||
|
}); |
||||||
|
}, |
||||||
|
} |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
$('.edit-project-board').each(function () { |
||||||
|
const projectTitleLabel = $(this).closest('.board-column-header').find('.board-label'); |
||||||
|
const projectTitleInput = $(this).find( |
||||||
|
'.content > .form > .field > .project-board-title' |
||||||
|
); |
||||||
|
|
||||||
|
$(this) |
||||||
|
.find('.content > .form > .actions > .red') |
||||||
|
.on('click', function (e) { |
||||||
|
e.preventDefault(); |
||||||
|
|
||||||
|
$.ajax({ |
||||||
|
url: $(this).data('url'), |
||||||
|
data: JSON.stringify({title: projectTitleInput.val()}), |
||||||
|
headers: { |
||||||
|
'X-Csrf-Token': csrf, |
||||||
|
'X-Remote': true, |
||||||
|
}, |
||||||
|
contentType: 'application/json', |
||||||
|
method: 'PUT', |
||||||
|
}).done(() => { |
||||||
|
projectTitleLabel.text(projectTitleInput.val()); |
||||||
|
projectTitleInput.closest('form').removeClass('dirty'); |
||||||
|
$('.ui.modal').modal('hide'); |
||||||
|
}); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
$('.delete-project-board').each(function () { |
||||||
|
$(this).click(function (e) { |
||||||
|
e.preventDefault(); |
||||||
|
|
||||||
|
$.ajax({ |
||||||
|
url: $(this).data('url'), |
||||||
|
headers: { |
||||||
|
'X-Csrf-Token': csrf, |
||||||
|
'X-Remote': true, |
||||||
|
}, |
||||||
|
contentType: 'application/json', |
||||||
|
method: 'DELETE', |
||||||
|
}).done(() => { |
||||||
|
setTimeout(window.location.reload(true), 2000); |
||||||
|
}); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
$('#new_board_submit').click(function (e) { |
||||||
|
e.preventDefault(); |
||||||
|
|
||||||
|
const boardTitle = $('#new_board'); |
||||||
|
|
||||||
|
$.ajax({ |
||||||
|
url: $(this).data('url'), |
||||||
|
data: JSON.stringify({title: boardTitle.val()}), |
||||||
|
headers: { |
||||||
|
'X-Csrf-Token': csrf, |
||||||
|
'X-Remote': true, |
||||||
|
}, |
||||||
|
contentType: 'application/json', |
||||||
|
method: 'POST', |
||||||
|
}).done(() => { |
||||||
|
boardTitle.closest('form').removeClass('dirty'); |
||||||
|
setTimeout(window.location.reload(true), 2000); |
||||||
|
}); |
||||||
|
}); |
||||||
|
} |
Loading…
Reference in new issue