[Notice] Announcing the End of Demo Server [Read me]

merge with nforge:master
@e839b2a3d07e513e6945480159f658a7a84df71a
--- app/assets/stylesheets/less/_page.less
+++ app/assets/stylesheets/less/_page.less
... | ... | @@ -447,7 +447,7 @@ |
447 | 447 |
position: absolute; |
448 | 448 |
right: -27px;top: 23px; |
449 | 449 |
width : 18px !important; |
450 |
- height: 16px !important; /* 40px: for 2 button */ |
|
450 |
+ height: 40px !important; /* 40px: for 2 button */ |
|
451 | 451 |
padding:16px 3px 16px 3px; |
452 | 452 |
|
453 | 453 |
.side-menu { |
... | ... | @@ -1494,12 +1494,12 @@ |
1494 | 1494 |
position: relative; |
1495 | 1495 |
overflow: hidden; |
1496 | 1496 |
|
1497 |
- /*height:40px;*/ |
|
1498 |
- display:block; clear:both; |
|
1497 |
+ height:40px; display:block; clear:both; |
|
1499 | 1498 |
|
1500 | 1499 |
.file { |
1501 |
- position: absolute; z-index:2; cursor:pointer; |
|
1502 |
- top:0; left:-5px; width:100px; |
|
1500 |
+ position: absolute; z-index:2; |
|
1501 |
+ cursor:pointer; |
|
1502 |
+ left: -15px; |
|
1503 | 1503 |
.opacity(0); |
1504 | 1504 |
} |
1505 | 1505 |
|
... | ... | @@ -2579,11 +2579,11 @@ |
2579 | 2579 |
width: 115px; /*155px;*/ |
2580 | 2580 |
} |
2581 | 2581 |
} |
2582 |
- |
|
2582 |
+ |
|
2583 | 2583 |
.branches { |
2584 |
- .label { |
|
2584 |
+ .label { |
|
2585 | 2585 |
margin:0; padding:0 5px; margin-right:3px; |
2586 |
- |
|
2586 |
+ |
|
2587 | 2587 |
&.branch { background-color:@orange; } |
2588 | 2588 |
} |
2589 | 2589 |
} |
... | ... | @@ -2794,6 +2794,8 @@ |
2794 | 2794 |
/*padding:2px 4px; cursor:pointer;*/ |
2795 | 2795 |
width:10px; cursor:e-resize; |
2796 | 2796 |
height: 99.9%; |
2797 |
+ vertical-align: middle; |
|
2798 |
+ color: gray; |
|
2797 | 2799 |
border-left: 1px solid #ccc; |
2798 | 2800 |
background: #e8e8e8; |
2799 | 2801 |
background: -webkit-gradient(linear, left top, right top, color-stop(0%,#f5f5f5), color-stop(100%,#e9e9e9)); |
--- app/controllers/BoardApp.java
+++ app/controllers/BoardApp.java
... | ... | @@ -58,6 +58,9 @@ |
58 | 58 |
Form<SearchCondition> postParamForm = new Form<SearchCondition>(SearchCondition.class); |
59 | 59 |
SearchCondition searchCondition = postParamForm.bindFromRequest().get(); |
60 | 60 |
searchCondition.pageNum = pageNum - 1; |
61 |
+ if(searchCondition.orderBy.equals("id")) { |
|
62 |
+ searchCondition.orderBy = "createdDate"; |
|
63 |
+ } |
|
61 | 64 |
|
62 | 65 |
ExpressionList<Posting> el = searchCondition.asExpressionList(project); |
63 | 66 |
Page<Posting> posts = el.findPagingList(ITEMS_PER_PAGE).getPage(searchCondition.pageNum); |
--- app/controllers/UserApp.java
+++ app/controllers/UserApp.java
... | ... | @@ -25,6 +25,8 @@ |
25 | 25 |
import java.util.ArrayList; |
26 | 26 |
import java.util.List; |
27 | 27 |
|
28 |
+import javax.persistence.Transient; |
|
29 |
+ |
|
28 | 30 |
import static play.data.Form.form; |
29 | 31 |
import static play.libs.Json.toJson; |
30 | 32 |
|
... | ... | @@ -198,6 +200,40 @@ |
198 | 200 |
return user; |
199 | 201 |
} |
200 | 202 |
|
203 |
+ public static Result resetUserPassword() { |
|
204 |
+ Form<User> userForm = form(User.class).bindFromRequest(); |
|
205 |
+ |
|
206 |
+ if(userForm.hasErrors()) { |
|
207 |
+ return badRequest(); |
|
208 |
+ } |
|
209 |
+ |
|
210 |
+ User currentUser = currentUser(); |
|
211 |
+ User user = userForm.get(); |
|
212 |
+ |
|
213 |
+ if(!isValidPassword(currentUser, user.oldPassword)) { |
|
214 |
+ Form<User> currentUserForm = new Form<User>(User.class); |
|
215 |
+ currentUserForm = currentUserForm.fill(currentUser); |
|
216 |
+ |
|
217 |
+ flash(Constants.WARNING, "user.wrongPassword.alert"); |
|
218 |
+ return badRequest(edit.render(currentUserForm, currentUser)); |
|
219 |
+ } |
|
220 |
+ |
|
221 |
+ resetPassword(currentUser, user.password); |
|
222 |
+ |
|
223 |
+ //go to login page |
|
224 |
+ session().clear(); |
|
225 |
+ response().discardCookie(TOKEN); |
|
226 |
+ |
|
227 |
+ flash(Constants.WARNING, "user.loginWithNewPassword"); |
|
228 |
+ return redirect(routes.UserApp.loginForm()); |
|
229 |
+ |
|
230 |
+ } |
|
231 |
+ |
|
232 |
+ public static boolean isValidPassword(User currentUser, String password) { |
|
233 |
+ String hashedOldPassword = hashedPassword(password, currentUser.passwordSalt); |
|
234 |
+ return currentUser.password.equals(hashedOldPassword); |
|
235 |
+ } |
|
236 |
+ |
|
201 | 237 |
public static void resetPassword(User user, String newPassword) { |
202 | 238 |
user.password = new Sha256Hash(newPassword, |
203 | 239 |
ByteSource.Util.bytes(user.passwordSalt), 1024).toBase64(); |
--- app/models/User.java
+++ app/models/User.java
... | ... | @@ -43,6 +43,9 @@ |
43 | 43 |
|
44 | 44 |
@Pattern(value = "^[a-zA-Z0-9-]+([_.][a-zA-Z0-9-]+)*$", message = "user.wrongloginId.alert") @Required |
45 | 45 |
public String loginId; |
46 |
+ |
|
47 |
+ @Transient |
|
48 |
+ public String oldPassword; |
|
46 | 49 |
public String password; |
47 | 50 |
public String passwordSalt; |
48 | 51 |
|
... | ... | @@ -175,7 +178,6 @@ |
175 | 178 |
ByteSource.Util.bytes(user.passwordSalt), 1024).toBase64(); |
176 | 179 |
user.save(); |
177 | 180 |
} |
178 |
- |
|
179 | 181 |
|
180 | 182 |
public Resource asResource() { |
181 | 183 |
return new Resource() { |
--- app/views/board/postList.scala.html
+++ app/views/board/postList.scala.html
... | ... | @@ -4,6 +4,8 @@ |
4 | 4 |
@import utils.AccessControl._ |
5 | 5 |
@import scala.collection.immutable.Map |
6 | 6 |
|
7 |
+@urlToList = {@routes.BoardApp.posts(project.owner, project.name, page.getPageIndex + 1)} |
|
8 |
+ |
|
7 | 9 |
@** header(label:String, key:String) = { |
8 | 10 |
<th> |
9 | 11 |
<a key="@key" href="@routes.BoardApp.posts(project.owner, project.name)">@label</a> |
... | ... | @@ -28,7 +30,7 @@ |
28 | 30 |
<form method="get" id="option_form"> |
29 | 31 |
<input type="hidden" name="orderBy" value="@param.orderBy"> |
30 | 32 |
<input type="hidden" name="orderDir" value="@param.orderDir"> |
31 |
- <input name="filter" class="text" type="text" placeholder="@Messages("project.searchPlaceholder")" value="@param.filter"><!-- |
|
33 |
+ <input name="filter" class="text" type="text" placeholder="@Messages("project.searchPlaceholder")" value="@param.filter"><!-- |
|
32 | 34 |
--><button type="submit" class="btn search-btn underConstruction">@Messages("post.menu.search")</button> |
33 | 35 |
</form> |
34 | 36 |
</div> |
... | ... | @@ -77,13 +79,25 @@ |
77 | 79 |
|
78 | 80 |
<div class="filter-wrap board"> |
79 | 81 |
<div class="filters" id="order"> |
80 |
- <a href="#" key="date" class="filter underConstruction @if(param.orderBy == "date"){ active }"> |
|
81 |
- <i class="ico btn-gray-arrow @if(param.orderBy == "date" && param.orderDir == "desc"){down}"></i>날짜순 |
|
82 |
- </a> |
|
82 |
+ @if(param.orderBy.equals("createdDate")) { |
|
83 |
+ @if(param.orderDir.equals("asc")) { |
|
84 |
+ <a href="#" data-orderBy="createdDate" data-orderDir="desc" class="filter active"><i class="ico btn-gray-arrow"></i>날짜순</a> |
|
85 |
+ } else { |
|
86 |
+ <a href="#" data-orderBy="createdDate" data-orderDir="asc" class="filter active"><i class="ico btn-gray-arrow down"></i>날짜순</a> |
|
87 |
+ } |
|
88 |
+ } else { |
|
89 |
+ <a href="#" data-orderBy="createdDate" data-orderDir="asc" class="filter"><i class="ico btn-gray-arrow"></i>날짜순</a> |
|
90 |
+ } |
|
83 | 91 |
|
84 |
- <a href="#" key="commentCount" class="filter underConstruction @if(param.orderBy == "commentCount"){ active }"> |
|
85 |
- <i class="ico btn-gray-arrow @if(param.orderBy == "commentCount" && param.orderDir == "desc"){down}"></i>댓글순 |
|
86 |
- </a> |
|
92 |
+ @if(param.orderBy.equals("numOfComments")) { |
|
93 |
+ @if(param.orderDir.equals("asc")) { |
|
94 |
+ <a href="#" data-orderBy="numOfComments" data-orderDir="desc" class="filter active"><i class="ico btn-gray-arrow"></i>댓글순</a> |
|
95 |
+ } else { |
|
96 |
+ <a href="#" data-orderBy="numOfComments" data-orderDir="asc" class="filter active"><i class="ico btn-gray-arrow down"></i>댓글순</a> |
|
97 |
+ } |
|
98 |
+ } else { |
|
99 |
+ <a href="#" data-orderBy="numOfComments" data-orderDir="asc" class="filter"><i class="ico btn-gray-arrow"></i>댓글순</a> |
|
100 |
+ } |
|
87 | 101 |
</div> |
88 | 102 |
</div> |
89 | 103 |
|
--- app/views/code/codeView.scala.html
+++ app/views/code/codeView.scala.html
... | ... | @@ -11,8 +11,8 @@ |
11 | 11 |
@makeBranchItem(branch:String) = { |
12 | 12 |
@defining(branch.split('/')){ names => |
13 | 13 |
@if(names(0).equals("refs") && names.length == 3){ |
14 |
- <li data-value="@branch"><a href="#"><!-- |
|
15 |
- --><span class="label @branchItemType(names(1))">@branchItemType(names(1))</span><!-- |
|
14 |
+ <li data-value="@branch"><a href="#"><!-- |
|
15 |
+ --><span class="label @branchItemType(names(1))">@branchItemType(names(1))</span><!-- |
|
16 | 16 |
-->@names(2) |
17 | 17 |
</a></li> |
18 | 18 |
} else { |
... | ... | @@ -40,7 +40,7 @@ |
40 | 40 |
<span class="clone-label"><strong class="@project.vcs">@project.vcs</strong>@Messages("code.repoUrl")</span><!-- |
41 | 41 |
--><input id="repositoryURL" type="text" class="text repo-url" readonly="readonly" value="@CodeApp.getURL(project.owner, project.name)"><!-- |
42 | 42 |
--><a id="copyURL" href="#!/copy-url" class="copy-btn btn">COPY TO CLIPBOARD</a> |
43 |
- |
|
43 |
+ |
|
44 | 44 |
<div id="branches" class="btn-group branches" data-name="branch" data-activate="manual"> |
45 | 45 |
<button data-toggle="dropdown" class="btn dropdown-toggle bgwhite d-label"></button> |
46 | 46 |
<button data-toggle="dropdown" class="btn dropdown-toggle bgwhite"><span class="caret"></span></button> |
... | ... | @@ -50,17 +50,6 @@ |
50 | 50 |
} |
51 | 51 |
</ul> |
52 | 52 |
</div> |
53 |
- <!-- |
|
54 |
- <div class="btn-group branches"> |
|
55 |
- <button data-toggle="dropdown" class="btn dropdown-toggle d-label bgwhite" id="selected-branch" data-branch="HEAD"></button> |
|
56 |
- <button data-toggle="dropdown" class="btn dropdown-toggle bgwhite"><span class="caret"></span></button> |
|
57 |
- <ul class="dropdown-menu"> |
|
58 |
- @for(name <- branches){ |
|
59 |
- <li><a href="#" class="branch-item" data-branch="@name">@name</a></li> |
|
60 |
- } |
|
61 |
- </ul> |
|
62 |
- </div> |
|
63 |
- --> |
|
64 | 53 |
</div> |
65 | 54 |
<hr class="double-sp" /> |
66 | 55 |
<div class="code-browse-wrap"> |
... | ... | @@ -71,7 +60,7 @@ |
71 | 60 |
</div> |
72 | 61 |
<div id="folderList" class="directories"></div> |
73 | 62 |
</div> |
74 |
- <div class="btnResize cell"></div> |
|
63 |
+ <div class="btnResize cell">|||</div> |
|
75 | 64 |
<div class="sp-wrap cell"></div> |
76 | 65 |
|
77 | 66 |
@** 파일 목록 **@ |
--- app/views/issue/issueList.scala.html
+++ app/views/issue/issueList.scala.html
... | ... | @@ -89,7 +89,7 @@ |
89 | 89 |
<div class="control-group"> |
90 | 90 |
<label class="control-label">@Messages("milestone")</label> |
91 | 91 |
<div class="controls"> |
92 |
- <div class="btn-group" data-name="milestone"> |
|
92 |
+ <div class="btn-group" data-name="milestoneId"> |
|
93 | 93 |
<button data-toggle="dropdown" class="btn dropdown-toggle bgwhite d-label input-medium">@Messages("milestone.state.all")</button> |
94 | 94 |
<button data-toggle="dropdown" class="btn dropdown-toggle bgwhite"><span class="caret"></span></button> |
95 | 95 |
<ul class="dropdown-menu"> |
--- app/views/user/edit.scala.html
+++ app/views/user/edit.scala.html
... | ... | @@ -34,7 +34,7 @@ |
34 | 34 |
@helper.inputText(userForm("email"),'_showConstraints -> false,'_label -> null, 'class -> "span2")</p> |
35 | 35 |
</div> |
36 | 36 |
|
37 |
- <input class="btn" type="submit" value="저장"/> |
|
37 |
+ <input class="btn" type="submit" value="@Messages("button.save")"/> |
|
38 | 38 |
} |
39 | 39 |
<hr/> |
40 | 40 |
<div class="user-other-info info-box"> |
... | ... | @@ -79,11 +79,34 @@ |
79 | 79 |
</li> |
80 | 80 |
} |
81 | 81 |
</ul> |
82 |
+ <div> |
|
83 |
+ <form name="passwordReset" action="@routes.UserApp.resetUserPassword()" method="post"> |
|
84 |
+ <input type="hidden" name="loginId" value="@user.loginId" /> |
|
85 |
+ <hr /> |
|
86 |
+ <dl> |
|
87 |
+ <dt>@Messages("user.password")</dt> |
|
88 |
+ <dd><input type="password" name="oldPassword" value="" autocomplete="off" /></dd> |
|
89 |
+ <dt>@Messages("user.newPassword")</dt> |
|
90 |
+ <dd><input type="password" id="password" name="password" value="" autocomplete="off" /></dd> |
|
91 |
+ <dt>@Messages("validation.retypePassword")</dt> |
|
92 |
+ <dd><input type="password" id="retypedPassword" name="retypedPassword" value="" autocomplete="off" /></dd> |
|
93 |
+ </dl> |
|
94 |
+ <input class="btn" type="submit" value="@Messages("button.save")" /> |
|
95 |
+ </form> |
|
96 |
+ <hr /> |
|
97 |
+ </div> |
|
82 | 98 |
</div> |
83 | 99 |
</section> |
84 | 100 |
</div> |
85 | 101 |
</div> |
86 | 102 |
|
103 |
+ <script type="text/javascript" src="@getJSLink("lib/validate")"></script> |
|
104 |
+ <script type="text/javascript"> |
|
105 |
+ $(document).ready(function(){ |
|
106 |
+ $hive.loadModule("user.ResetPassword"); |
|
107 |
+ }); |
|
108 |
+ </script> |
|
109 |
+ |
|
87 | 110 |
<script src="@getJSLink("lib/jquery/jquery.form")" type="text/javascript"></script> |
88 | 111 |
<script type="text/javascript"> |
89 | 112 |
(function(){ |
--- conf/messages.en
+++ conf/messages.en
... | ... | @@ -299,6 +299,7 @@ |
299 | 299 |
user.loginId.duplicate = Already existing ID |
300 | 300 |
user.login.alert = Please, login. |
301 | 301 |
user.password = Password |
302 |
+user.newPassword = New Password |
|
302 | 303 |
user.confirmPassword = Confirm Password |
303 | 304 |
user.confirmPassword.alert = Password doesn't match the confirmation |
304 | 305 |
user.wrongPassword.alert = Wrong password! |
--- conf/messages.ko
+++ conf/messages.ko
... | ... | @@ -305,6 +305,7 @@ |
305 | 305 |
user.loginId.duplicate = 이미 존재하는 아이디입니다. |
306 | 306 |
user.login.alert = 로그인이 필요합니다. |
307 | 307 |
user.password = 비밀번호 |
308 |
+user.newPassword = 신규 비밀번호 |
|
308 | 309 |
user.confirmPassword = 비밀번호 확인 |
309 | 310 |
user.confirmPassword.alert = 입력한 두 비밀번호가 서로 일치하지 않습니다 |
310 | 311 |
user.wrongPassword.alert = 잘못된 비밀번호입니다! |
--- conf/routes
+++ conf/routes
... | ... | @@ -23,6 +23,7 @@ |
23 | 23 |
POST /users/signup controllers.UserApp.newUser() |
24 | 24 |
GET /user/editform controllers.UserApp.editUserInfoForm() |
25 | 25 |
POST /user/edit controllers.UserApp.editUserInfo() |
26 |
+POST /user/resetPassword controllers.UserApp.resetUserPassword() |
|
26 | 27 |
GET /user/isExist/:loginId controllers.UserApp.isUserExist(loginId) |
27 | 28 |
GET /user/isEmailExist/:email controllers.UserApp.isEmailExist(email) |
28 | 29 |
|
--- public/javascripts/template/NamingRule.md
+++ docs/technical/javascript-naming-convention.md
... | ... | @@ -5,7 +5,7 @@ |
5 | 5 |
|
6 | 6 |
* 단어를 생략하거나 약어를 사용하지 않는다. 단, HTML,URL 등과 같은 범용적인 약어는 사용할 수 있다. 약어 사용시에는 모두 대문자로 작성한다. |
7 | 7 |
* 한글 발음을 로마자로 그대로 표기하지 않는다 |
8 |
-* 특수 문자는 가급적 사용하지 않는다. 단, 상수의 이름에서 단어를 구분하기 위한 용도와 Private 지시자 표시를 위하여 언더스코어(_)를 사용한다 |
|
8 |
+* 특수 문자는 가급적 사용하지 않는다. 단, 상수의 이름에서 단어를 구분하기 위한 용도와 Private 지시자 표시를 위하여 언더스코어(`_`)를 사용한다 |
|
9 | 9 |
* 2글자 이상 대문자를 연속해서 사용하지 않는다. 단, 상수 이름이나 약어는 예외로 한다 |
10 | 10 |
* 상수 이름이나 약어 이름을 대문자를 사용한다 |
11 | 11 |
* 이름을 통해 역할과 목적을 할 수 있도록 간결하고 명료하게 작성한다 |
... | ... | @@ -68,16 +68,16 @@ |
68 | 68 |
> getElement(); |
69 | 69 |
> showLayer(); |
70 | 70 |
|
71 |
-* Private 메소드 인 경우 메소드 이름 앞에 언더스코어(_)를 사용한다 |
|
71 |
+* Private 메소드 인 경우 메소드 이름 앞에 언더스코어(`_`)를 사용한다 |
|
72 | 72 |
* 카멜 표기법을 준수한다. 복합어 이름은 첫 번째 단어를 소문자로 작성하고, 두 번째 이상의 단어 첫 글자를 대문자로 작성하여 단어를 구분한다. |
73 |
-* 함수 이름의 첫 글자로 연속된 2개의 언더스코어(__) 기호와 달러 기호($)는 사용하지 않는다. |
|
73 |
+* 함수 이름의 첫 글자로 연속된 2개의 언더스코어(`__`) 기호와 달러 기호($)는 사용하지 않는다. |
|
74 | 74 |
* Getter, Setter 메소드는 반드시 'get + 멤버변수 이름', 'set + 멤버변수 이름' 형식으로 작성한다. 단, Getter 메소드의 반환값이 Boolean 인 경우 get 대신 is 조합을 사용한다 |
75 | 75 |
|
76 | 76 |
> getElement(); |
77 | 77 |
> isChecked(); |
78 | 78 |
> setOption(); |
79 | 79 |
|
80 |
-* 이벤트 핸들러 메소드는 _on + 이벤트명 으로 정의한다. 브라우저에서 제공하는 이벤트가 아니라 특정 모듈이나 함수에서 비동기 콜백 함수(이벤트) 모델을 사용하는 경우에도 동일하다. |
|
80 |
+* 이벤트 핸들러 메소드는 `_on` + 이벤트명 으로 정의한다. 브라우저에서 제공하는 이벤트가 아니라 특정 모듈이나 함수에서 비동기 콜백 함수(이벤트) 모델을 사용하는 경우에도 동일하다. |
|
81 | 81 |
|
82 | 82 |
> function _onLoadImage(){ |
83 | 83 |
> console.log("image loaded"); |
... | ... | @@ -94,7 +94,7 @@ |
94 | 94 |
## 변수 이름 |
95 | 95 |
|
96 | 96 |
* 변수 이름은 명사를 사용하여 작성한다 |
97 |
-* Private 프로퍼티일 경우 변수 이름 앞에 언더스코어(_)를 사용한다 |
|
97 |
+* Private 프로퍼티일 경우 변수 이름 앞에 언더스코어(`_`)를 사용한다 |
|
98 | 98 |
* 헝가리안 표기법을 준수한다. 변수의 데이터 타입, 용도에 따라 접두사를 사용한다. jQuery 프레임워크에 의해 래핑(Wrapping) 된 변수는 접두사 w를 다른 접두사와 함께 사용한다 |
99 | 99 |
|
100 | 100 |
>g: 전역변수 |
... | ... | @@ -169,7 +169,7 @@ |
169 | 169 |
* 연산자의 앞과 뒤에는 공백을 삽입한다. 단, 단항 연산자, 전위/후위 연산자는 예외로 한다. |
170 | 170 |
* 소괄호( ( ) )와 키워드(if, for, return)를 사용할 때 공백을 삽입하지 않는다 |
171 | 171 |
* 시작 중괄호({)는 앞에 공백을 삽입하지 않는다 |
172 |
-* 함수 선언 직후에는 빈 줄을 삽입하지 않는다 (function _method(){ 바로 다음 줄) |
|
172 |
+* 함수 선언 직후에는 빈 줄을 삽입하지 않는다 (`function _method(){` 바로 다음 줄) |
|
173 | 173 |
* 함수 선언 사이에는 빈 줄을 삽입한다 (함수와 함수 사이) |
174 | 174 |
* 변수를 선언한 경우에는 다음에 빈 줄을 삽입하지 않는다. |
175 | 175 |
* 명령문 간에는 빈 줄을 사용하지 않는다. 단, 소스 코드의 길이가 길어지는 경우 구분을 위해 삽입할 수 있다. |
--- public/javascripts/common/hive.ui.Dropdown.js
+++ public/javascripts/common/hive.ui.Dropdown.js
... | ... | @@ -116,7 +116,7 @@ |
116 | 116 |
htVar.fOnChange = fOnChange; |
117 | 117 |
return true; |
118 | 118 |
} |
119 |
- |
|
119 |
+ |
|
120 | 120 |
/** |
121 | 121 |
* 현재 선택된 값을 반환 |
122 | 122 |
* @return {String} |
... | ... | @@ -124,14 +124,14 @@ |
124 | 124 |
function _getValue(){ |
125 | 125 |
return htVar.sValue; |
126 | 126 |
} |
127 |
- |
|
127 |
+ |
|
128 | 128 |
/** |
129 | 129 |
* 기본값 지정이 있으면 선택된 상태로 만들기 |
130 | 130 |
*/ |
131 | 131 |
function _selectDefault(){ |
132 | 132 |
return _selectItem("li[data-selected=true]"); |
133 | 133 |
} |
134 |
- |
|
134 |
+ |
|
135 | 135 |
/** |
136 | 136 |
* 지정한 값을 data-value 로 가진 항목을 선택 상태로 만듬 |
137 | 137 |
* @param {String} sValue |
... | ... | @@ -139,7 +139,7 @@ |
139 | 139 |
function _selectByValue(sValue){ |
140 | 140 |
return _selectItem("li[data-value='" + sValue + "']"); |
141 | 141 |
} |
142 |
- |
|
142 |
+ |
|
143 | 143 |
/** |
144 | 144 |
* 지정한 항목을 선택 상태로 만듬 |
145 | 145 |
* @param {String} sQuery 항목 선택 셀렉터 구문 |
... | ... | @@ -153,12 +153,12 @@ |
153 | 153 |
var welTarget = $(waFind[0]); |
154 | 154 |
_setItemSelected(welTarget); |
155 | 155 |
_setFormValue(welTarget); |
156 |
- |
|
156 |
+ |
|
157 | 157 |
return true; |
158 | 158 |
} |
159 | 159 |
|
160 | 160 |
_init(htOptions); |
161 |
- |
|
161 |
+ |
|
162 | 162 |
return { |
163 | 163 |
"getValue": _getValue, |
164 | 164 |
"onChange": _setOnChange, |
--- public/javascripts/service/hive.board.List.js
+++ public/javascripts/service/hive.board.List.js
... | ... | @@ -3,15 +3,15 @@ |
3 | 3 |
* |
4 | 4 |
* Copyright NHN Corporation. |
5 | 5 |
* Released under the MIT license |
6 |
- * |
|
6 |
+ * |
|
7 | 7 |
* http://hive.dev.naver.com/license |
8 | 8 |
*/ |
9 | 9 |
|
10 | 10 |
(function(ns){ |
11 |
- |
|
11 |
+ |
|
12 | 12 |
var oNS = $hive.createNamespace(ns); |
13 | 13 |
oNS.container[oNS.name] = function(htOptions){ |
14 |
- |
|
14 |
+ |
|
15 | 15 |
var htElement = {}; |
16 | 16 |
var htOrderMap = {"asc": "desc", "desc": "asc"}; |
17 | 17 |
|
... | ... | @@ -22,7 +22,7 @@ |
22 | 22 |
function _init(htOptions){ |
23 | 23 |
_initElement(htOptions || {}); |
24 | 24 |
_attachEvent(); |
25 |
- |
|
25 |
+ |
|
26 | 26 |
_initPagination(htOptions); |
27 | 27 |
} |
28 | 28 |
|
... | ... | @@ -31,12 +31,12 @@ |
31 | 31 |
*/ |
32 | 32 |
function _initElement(htOptions){ |
33 | 33 |
htElement.welForm = $(htOptions.sOptionForm || "#option_form"); |
34 |
- htElement.welInputKey = htElement.welForm.find("input[name=key]"); |
|
35 |
- htElement.welInputOrder = htElement.welForm.find("input[name=order]"); |
|
34 |
+ htElement.welInputOrderBy = htElement.welForm.find("input[name=orderBy]"); |
|
35 |
+ htElement.welInputOrderDir = htElement.welForm.find("input[name=orderDir]"); |
|
36 | 36 |
htElement.welInputPageNum = htElement.welForm.find("input[name=pageNum]"); |
37 |
- |
|
37 |
+ |
|
38 | 38 |
htElement.welFilter = $(htOptions.sQueryFilter || "#order a"); |
39 |
- htElement.welPages = $(htOptions.sQueryPages || "#pagination a"); |
|
39 |
+ htElement.welPages = $(htOptions.sQueryPages || "#pagination a"); |
|
40 | 40 |
htElement.welPagination = $(htOptions.elPagination || '#pagination'); |
41 | 41 |
} |
42 | 42 |
|
... | ... | @@ -52,15 +52,12 @@ |
52 | 52 |
* onClick filter |
53 | 53 |
*/ |
54 | 54 |
function _onClickFilter(){ |
55 |
- var sKey = $(this).attr("key"); |
|
55 |
+ var orderBy = $(this).attr("data-orderBy"); |
|
56 |
+ var orderDir = $(this).attr("data-orderDir"); |
|
56 | 57 |
|
57 |
- // Key |
|
58 |
- if (sKey !== htElement.welInputKey.val()) { |
|
59 |
- htElement.welInputKey.val(sKey) |
|
60 |
- } else { // Order |
|
61 |
- var sCurrentVal = htElement.welInputOrder.val(); |
|
62 |
- htElement.welInputOrder.val(htOrderMap[sCurrentVal]); |
|
63 |
- } |
|
58 |
+ htElement.welInputOrderBy.val(orderBy); |
|
59 |
+ htElement.welInputOrderDir.val(orderDir); |
|
60 |
+ |
|
64 | 61 |
htElement.welForm.submit(); |
65 | 62 |
return false; |
66 | 63 |
} |
... | ... | @@ -73,58 +70,13 @@ |
73 | 70 |
htElement.welForm.submit(); |
74 | 71 |
return false; |
75 | 72 |
} |
76 |
- |
|
77 |
- |
|
73 |
+ |
|
74 |
+ |
|
78 | 75 |
function _initPagination(htOptions){ |
79 | 76 |
hive.Pagination.update(htElement.welPagination, htOptions.nTotalPages); |
80 | 77 |
} |
81 |
- |
|
78 |
+ |
|
82 | 79 |
_init(htOptions); |
83 | 80 |
}; |
84 |
- |
|
85 |
-})("hive.board.List"); |
|
86 | 81 |
|
87 |
-/* |
|
88 |
-nforge.namespace("board"); |
|
89 |
-nforge.board.list = function() { |
|
90 |
- var that = { |
|
91 |
- "init" : function() { |
|
92 |
- that.setUpEventListener(); |
|
93 |
- }, |
|
94 |
- |
|
95 |
- "setUpEventListener" : function() { |
|
96 |
- var $headers = $("#order a"); |
|
97 |
- $headers.click(that.onHeader); |
|
98 |
- |
|
99 |
- var $pagination = $("#pagination a"); |
|
100 |
- $pagination.click(that.onPager); |
|
101 |
- }, |
|
102 |
- |
|
103 |
- "onHeader" : function() { |
|
104 |
- var key = $(this).attr("key"); |
|
105 |
- var $input = $("#option_form input[name=key]"); |
|
106 |
- if (key !== $input.val()) { |
|
107 |
- $input.val(key) |
|
108 |
- } else { |
|
109 |
- $input = $("#option_form input[name=order]"); |
|
110 |
- if ($input.val() === "desc"){ |
|
111 |
- $input.val("asc"); |
|
112 |
- } else if ($input.val() === "asc") { |
|
113 |
- $input.val("desc"); |
|
114 |
- } |
|
115 |
- } |
|
116 |
- $("#option_form").submit(); |
|
117 |
- return false; |
|
118 |
- }, |
|
119 |
- |
|
120 |
- "onPager" : function() { |
|
121 |
- var $input = $("#option_form input[name=pageNum]"); |
|
122 |
- $input.val($(this).attr("pageNum")); |
|
123 |
- $("#option_form").submit(); |
|
124 |
- return false; |
|
125 |
- } |
|
126 |
- }; |
|
127 |
- |
|
128 |
- return that; |
|
129 |
-}; |
|
130 |
-*/(파일 끝에 줄바꿈 문자 없음) |
|
82 |
+})("hive.board.List");(파일 끝에 줄바꿈 문자 없음) |
--- public/javascripts/service/hive.code.Browser.js
+++ public/javascripts/service/hive.code.Browser.js
... | ... | @@ -16,7 +16,7 @@ |
16 | 16 |
var oBranch = new hive.ui.Dropdown({ |
17 | 17 |
"elContainer": $("#branches") |
18 | 18 |
}); |
19 |
- |
|
19 |
+ |
|
20 | 20 |
$(document).ready(function(){ |
21 | 21 |
$(window).bind('hashchange', function(e){ |
22 | 22 |
// _updateDynaTree(); |
... | ... | @@ -201,32 +201,37 @@ |
201 | 201 |
function getBranch(){ |
202 | 202 |
return encodeURIComponent(oBranch.getValue()); |
203 | 203 |
} |
204 |
- |
|
204 |
+ |
|
205 | 205 |
/** resize list **/ |
206 | 206 |
function _initResizeList(){ |
207 | 207 |
var nFolderListX = $("#folderList").offset().left; |
208 | 208 |
var welBtnResize = $(".btnResize"); |
209 | 209 |
var welWrapDirectory = $(".directory-wrap"); |
210 | 210 |
var waWrapFile = $(".file-wrap"); // fileList, fileView |
211 |
- |
|
212 |
- welBtnResize.mousedown(function(){ |
|
213 |
- $(window).bind("mousemove", _resizeList); |
|
214 |
- return false; |
|
215 |
- }); |
|
216 |
- welBtnResize.mouseup(function(){ |
|
211 |
+ var draggable = true; |
|
212 |
+ |
|
213 |
+ welBtnResize.mousedown(function () { |
|
214 |
+ if(draggable) { |
|
215 |
+ $(window).bind("mousemove", _resizeList); |
|
216 |
+ } |
|
217 |
+ return false; |
|
218 |
+ }); |
|
219 |
+ $(".directory-wrap").mouseup(function(){ |
|
217 | 220 |
$(window).unbind("mousemove", _resizeList); |
218 | 221 |
return false; |
219 | 222 |
}); |
220 | 223 |
$(window).click(function(){ // for IE |
221 | 224 |
$(window).unbind("mousemove", _resizeList); |
222 | 225 |
}); |
223 |
- |
|
226 |
+ |
|
224 | 227 |
// 더블클릭하면 디렉토리 목록 숨김 |
225 | 228 |
welBtnResize.dblclick(function(){ |
226 | 229 |
if(welWrapDirectory.css("display") == "none"){ |
230 |
+ draggable = true; |
|
227 | 231 |
welWrapDirectory.show(); |
228 | 232 |
waWrapFile.width(850 - welWrapDirectory.width()); |
229 | 233 |
} else { |
234 |
+ draggable = false; |
|
230 | 235 |
welWrapDirectory.hide(); |
231 | 236 |
waWrapFile.width(850); |
232 | 237 |
} |
... | ... | @@ -235,6 +240,7 @@ |
235 | 240 |
function _resizeList(weEvt){ |
236 | 241 |
var nWidth = weEvt.clientX - nFolderListX; |
237 | 242 |
$(".directory-wrap").width(nWidth); |
243 |
+ $(".directories").width(nWidth); |
|
238 | 244 |
$(".file-wrap").width(850 - nWidth); |
239 | 245 |
} |
240 | 246 |
} |
+++ public/javascripts/service/hive.user.ResetPassword.js
... | ... | @@ -0,0 +1,134 @@ |
1 | +/** | |
2 | + * @(#)hive.user.Reset.js 2013.04.25 | |
3 | + * | |
4 | + * Copyright NHN Corporation. | |
5 | + * Released under the MIT license | |
6 | + * | |
7 | + * http://hive.dev.naver.com/license | |
8 | + */ | |
9 | +(function(ns){ | |
10 | + | |
11 | + var oNS = $hive.createNamespace(ns); | |
12 | + oNS.container[oNS.name] = function(){ | |
13 | + | |
14 | + var htVar = {}; | |
15 | + var htElement = {}; | |
16 | + | |
17 | + /** | |
18 | + * initialize | |
19 | + */ | |
20 | + function _init(){ | |
21 | + _initElement(); | |
22 | + _initFormValidator(); | |
23 | + _attachEvent(); | |
24 | + } | |
25 | + | |
26 | + /** | |
27 | + * initialize elements | |
28 | + */ | |
29 | + function _initElement(){ | |
30 | + htElement.welInputOldPassword = $('#oldPassword'); | |
31 | + htElement.welInputPassword = $('#password'); | |
32 | + htElement.welInputRetypedPassword = $('#retypedPassword'); | |
33 | + | |
34 | + htElement.welForm = $("form[name=passwordReset]"); | |
35 | + } | |
36 | + | |
37 | + /** | |
38 | + * attach event | |
39 | + */ | |
40 | + function _attachEvent(){ | |
41 | + htElement.welInputOldPassword.focusout(_onBlurInputPassword); | |
42 | + htElement.welInputPassword.focusout(_onBlurInputPassword); | |
43 | + htElement.welInputRetypedPassword.focusout(_onBlurInputPassword); | |
44 | + } | |
45 | + | |
46 | + | |
47 | + /** | |
48 | + * 비밀번호 확인 입력란 벗어날 때 이벤트 핸들러 | |
49 | + * 마지막 입력란이므로 전체 폼 유효성 검사 | |
50 | + */ | |
51 | + function _onBlurInputPassword(){ | |
52 | + htVar.oValidator._validateForm(); | |
53 | + } | |
54 | + | |
55 | + /** | |
56 | + * initialize FormValidator | |
57 | + * @require validate.js | |
58 | + */ | |
59 | + function _initFormValidator(){ | |
60 | + var aRules = [ | |
61 | + {"name": 'oldPassword', "rules": 'required'}, | |
62 | + {"name": 'password', "rules": 'required|min_length[4]'}, | |
63 | + {"name": 'retypedPassword', "rules": 'required|matches[password]'} | |
64 | + ]; | |
65 | + | |
66 | + htVar.oValidator = new FormValidator('passwordReset', aRules, _onFormValidate); | |
67 | + | |
68 | + // set error message | |
69 | + htVar.oValidator.setMessage('required', Messages("validation.required")); | |
70 | + htVar.oValidator.setMessage('min_length', Messages("validation.tooShortPassword")); | |
71 | + htVar.oValidator.setMessage('matches', Messages("validation.passwordMismatch")); | |
72 | + } | |
73 | + | |
74 | + /** | |
75 | + * on validate form | |
76 | + * @param {Array} aErrors | |
77 | + */ | |
78 | + function _onFormValidate(aErrors){ | |
79 | + _clearTooltips(); | |
80 | + // to avoid bootstrap bug | |
81 | + if (aErrors.length <= 0) { | |
82 | + return _clearTooltips(); | |
83 | + } | |
84 | + | |
85 | + var welTarget; | |
86 | + aErrors.forEach(function(htError){ | |
87 | + welTarget = htElement.welForm.find("input[name=" + htError.name + "]"); | |
88 | + if(welTarget){ | |
89 | + showErrorMessage(welTarget, htError.message); | |
90 | + } | |
91 | + }); | |
92 | + } | |
93 | + | |
94 | + /** | |
95 | + * 폼 영역에 있는 jquery.tooltip 모두 제거하는 함수 | |
96 | + */ | |
97 | + function _clearTooltips(){ | |
98 | + try { | |
99 | + htElement.welForm.find("input").each(function(i, v){ | |
100 | + $(v).tooltip("destroy"); | |
101 | + }); | |
102 | + } catch(e){} | |
103 | + } | |
104 | + | |
105 | + /** | |
106 | + * Bootstrap toolTip function has some limitation. | |
107 | + * In this case, toolTip doesn't provide easy way to change title and contents. | |
108 | + * So, unfortunately I had to change data value in directly. | |
109 | + * @param {Wrapped Element} welInput | |
110 | + * @param {String} sMessage | |
111 | + */ | |
112 | + function showErrorMessage(welInput, sMessage){ | |
113 | + welInput.tooltip({"trigger": "manual", "placement": "right"}); | |
114 | + | |
115 | + var oToolTip = welInput.data('tooltip'); | |
116 | + oToolTip.options.placement = 'right'; | |
117 | + oToolTip.options.trigger = 'manual'; | |
118 | + oToolTip.options.title = sMessage; | |
119 | + oToolTip.options.content = sMessage; | |
120 | + | |
121 | + welInput.tooltip('show'); | |
122 | + } | |
123 | + | |
124 | + function hideErrorMessage(welInput){ | |
125 | + welInput.tooltip("hide"); | |
126 | + | |
127 | + try{ | |
128 | + welInput.tooltip("destroy"); | |
129 | + } catch(e){} // to avoid bootstrap bug | |
130 | + } | |
131 | + | |
132 | + _init(); | |
133 | + }; | |
134 | +})("hive.user.ResetPassword");(파일 끝에 줄바꿈 문자 없음) |
--- public/javascripts/template/ModuleGuide.md
+++ public/javascripts/template/ModuleGuide.md
... | ... | @@ -4,7 +4,7 @@ |
4 | 4 |
|
5 | 5 |
---- |
6 | 6 |
## 명명 규칙 |
7 |
- |
|
7 |
+ |
|
8 | 8 |
* NamingGuide.md 참조 |
9 | 9 |
|
10 | 10 |
## 폴더 구조 |
... | ... | @@ -19,7 +19,7 @@ |
19 | 19 |
|
20 | 20 |
* __javascripts/service__ |
21 | 21 |
|
22 |
- 각 페이지에 사용되는 자바스크립트 모듈 |
|
22 |
+ 각 페이지에 사용되는 자바스크립트 모듈. |
|
23 | 23 |
함수로 작성한다 |
24 | 24 |
|
25 | 25 |
* __javascripts/deprecated__ |
... | ... | @@ -50,24 +50,23 @@ |
50 | 50 |
* 현재 페이지 내에서 해당 모듈을 찾을 수 없는 경우, 즉 명시적으로 <script> 태그를 통해 포함하지 않은 경우 |
51 | 51 |
자동으로 javascripts/service 에서 동적으로 자바스크립트 파일을 로딩하려 시도한다. |
52 | 52 |
이미 페이지 내에 <script> 태그를 이용해 포함한 경우에는 동적 로딩은 시도되지 않는다. |
53 |
- 동적 로딩을 시도하는 파일 경로는 javascripts/service/hive.(module.Name).js 이다. |
|
53 |
+ 동적 로딩을 시도하는 파일 경로는 javascripts/service/hive.(module.Name).js 이다. |
|
54 | 54 |
* 자바스크립트 파일 로딩이 완료되어 모듈 코드를 사용할 수 있을 때 자동으로 초기화를 시도하며 내부적으로 수행되는 코드는 아래와 같다. |
55 | 55 |
이 중 htOption 변수는 $hive.loadModule() 의 두번째 인자와 동일하다 |
56 | 56 |
|
57 | 57 |
> new hive.module.Name(htOption) |
58 | 58 |
|
59 |
-* $hive.loadModule()는 모듈 함수를 실행하는 역할만 한다. 별도의 인터페이스가 필요한 것은 아니기 때문에 |
|
59 |
+* $hive.loadModule()는 모듈 함수를 실행하는 역할만 한다. 별도의 인터페이스가 필요한 것은 아니기 때문에 |
|
60 | 60 |
모듈 내부의 함수 구조가 모듈 로딩 자체에 영향을 주지는 않는다. |
61 | 61 |
|
62 | 62 |
## 기타 |
63 | 63 |
|
64 |
-* HTML 템플릿은 최대한 자바스크립트 파일 내에 포함하지 않는다 |
|
65 |
-* HTML 템플릿 데이터는 정적 페이지내에 <script type="text/template"> 형태로 위치시키고 자바스크립트는 그 내용을 활용하는 형태로 작성한다 |
|
64 |
+* HTML 템플릿은 가능한 한 자바스크립트 파일 내에 포함하지 않는다 |
|
65 |
+* HTML 템플릿 데이터는 정적 페이지내에 <script type="text/template"> 형태로 위치시키고 자바스크립트는 그 내용을 활용하는 형태로 작성한다 |
|
66 | 66 |
|
67 | 67 |
> <script type="text/template" id="tplItem"> |
68 | 68 |
> ${name} ${email} |
69 | 69 |
> </script> |
70 | 70 |
> |
71 | 71 |
> var sTpl = document.getElementById("tplItem").text; |
72 |
- |
|
73 | 72 |
|
--- public/javascripts/template/hive.CodeTemplate.js
+++ public/javascripts/template/hive.CodeTemplate.js
... | ... | @@ -65,7 +65,11 @@ |
65 | 65 |
// detachEvent() if available |
66 | 66 |
|
67 | 67 |
// free memory |
68 |
+<<<<<<< HEAD |
|
68 | 69 |
htVar = htElement = null; |
70 |
+======= |
|
71 |
+ htVar = htElement = null; |
|
72 |
+>>>>>>> 3283b2ee5c73643ae8db9a489d502b3ed40d2cca |
|
69 | 73 |
} |
70 | 74 |
|
71 | 75 |
_init(); |
Add a comment
Delete comment
Once you delete this comment, you won't be able to recover it. Are you sure you want to delete this comment?