You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

UserPage.java 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. /*
  2. * Copyright 2012 gitblit.com.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.gitblit.wicket.pages;
  17. import java.io.Serializable;
  18. import java.util.ArrayList;
  19. import java.util.Arrays;
  20. import java.util.Collections;
  21. import java.util.Comparator;
  22. import java.util.List;
  23. import java.util.Locale;
  24. import org.apache.wicket.PageParameters;
  25. import org.apache.wicket.ajax.AjaxRequestTarget;
  26. import org.apache.wicket.ajax.markup.html.form.AjaxButton;
  27. import org.apache.wicket.markup.html.basic.Label;
  28. import org.apache.wicket.markup.html.form.Form;
  29. import org.apache.wicket.markup.html.panel.Fragment;
  30. import org.apache.wicket.markup.repeater.Item;
  31. import org.apache.wicket.markup.repeater.data.DataView;
  32. import org.apache.wicket.markup.repeater.data.ListDataProvider;
  33. import org.apache.wicket.model.IModel;
  34. import org.apache.wicket.model.Model;
  35. import com.gitblit.Constants.Transport;
  36. import com.gitblit.GitBlitException;
  37. import com.gitblit.Keys;
  38. import com.gitblit.models.Menu.ParameterMenuItem;
  39. import com.gitblit.models.NavLink;
  40. import com.gitblit.models.NavLink.DropDownPageMenuNavLink;
  41. import com.gitblit.models.ProjectModel;
  42. import com.gitblit.models.RepositoryModel;
  43. import com.gitblit.models.UserModel;
  44. import com.gitblit.utils.StringUtils;
  45. import com.gitblit.wicket.GitBlitWebApp;
  46. import com.gitblit.wicket.GitBlitWebSession;
  47. import com.gitblit.wicket.GitblitRedirectException;
  48. import com.gitblit.wicket.WicketUtils;
  49. import com.gitblit.wicket.panels.BooleanOption;
  50. import com.gitblit.wicket.panels.ChoiceOption;
  51. import com.gitblit.wicket.panels.ProjectRepositoryPanel;
  52. import com.gitblit.wicket.panels.SshKeysPanel;
  53. import com.gitblit.wicket.panels.TextOption;
  54. import com.gitblit.wicket.panels.UserTitlePanel;
  55. public class UserPage extends RootPage {
  56. List<ProjectModel> projectModels = new ArrayList<ProjectModel>();
  57. public UserPage() {
  58. super();
  59. throw new GitblitRedirectException(GitBlitWebApp.get().getHomePage());
  60. }
  61. public UserPage(PageParameters params) {
  62. super(params);
  63. setup(params);
  64. }
  65. @Override
  66. protected boolean reusePageParameters() {
  67. return true;
  68. }
  69. private void setup(PageParameters params) {
  70. setupPage("", "");
  71. // check to see if we should display a login message
  72. boolean authenticateView = app().settings().getBoolean(Keys.web.authenticateViewPages, true);
  73. if (authenticateView && !GitBlitWebSession.get().isLoggedIn()) {
  74. authenticationError("Please login");
  75. return;
  76. }
  77. String userName = WicketUtils.getUsername(params);
  78. if (StringUtils.isEmpty(userName)) {
  79. throw new GitblitRedirectException(GitBlitWebApp.get().getHomePage());
  80. }
  81. UserModel user = app().users().getUserModel(userName);
  82. if (user == null) {
  83. // construct a temporary user model
  84. user = new UserModel(userName);
  85. }
  86. add(new UserTitlePanel("userTitlePanel", user, user.username));
  87. UserModel sessionUser = GitBlitWebSession.get().getUser();
  88. boolean isMyProfile = sessionUser != null && sessionUser.equals(user);
  89. if (isMyProfile) {
  90. addPreferences(user);
  91. if (app().gitblit().isServingSSH()) {
  92. // show the SSH key management tab
  93. addSshKeys(user);
  94. } else {
  95. // SSH daemon is disabled, hide keys tab
  96. add(new Label("sshKeysLink").setVisible(false));
  97. add(new Label("sshKeysTab").setVisible(false));
  98. }
  99. } else {
  100. // visiting user
  101. add(new Label("preferencesLink").setVisible(false));
  102. add(new Label("preferencesTab").setVisible(false));
  103. add(new Label("sshKeysLink").setVisible(false));
  104. add(new Label("sshKeysTab").setVisible(false));
  105. }
  106. List<RepositoryModel> repositories = getRepositories(params);
  107. Collections.sort(repositories, new Comparator<RepositoryModel>() {
  108. @Override
  109. public int compare(RepositoryModel o1, RepositoryModel o2) {
  110. // reverse-chronological sort
  111. return o2.lastChange.compareTo(o1.lastChange);
  112. }
  113. });
  114. final ListDataProvider<RepositoryModel> dp = new ListDataProvider<RepositoryModel>(repositories);
  115. DataView<RepositoryModel> dataView = new DataView<RepositoryModel>("repositoryList", dp) {
  116. private static final long serialVersionUID = 1L;
  117. @Override
  118. public void populateItem(final Item<RepositoryModel> item) {
  119. final RepositoryModel entry = item.getModelObject();
  120. ProjectRepositoryPanel row = new ProjectRepositoryPanel("repository",
  121. getLocalizer(), this, showAdmin, entry, getAccessRestrictions());
  122. item.add(row);
  123. }
  124. };
  125. add(dataView);
  126. }
  127. @Override
  128. protected void addDropDownMenus(List<NavLink> navLinks) {
  129. PageParameters params = getPageParameters();
  130. DropDownPageMenuNavLink menu = new DropDownPageMenuNavLink("gb.filters",
  131. UserPage.class);
  132. // preserve time filter option on repository choices
  133. menu.menuItems.addAll(getRepositoryFilterItems(params));
  134. // preserve repository filter option on time choices
  135. menu.menuItems.addAll(getTimeFilterItems(params));
  136. if (menu.menuItems.size() > 0) {
  137. // Reset Filter
  138. menu.menuItems.add(new ParameterMenuItem(getString("gb.reset")));
  139. }
  140. navLinks.add(menu);
  141. }
  142. private void addPreferences(UserModel user) {
  143. // add preferences
  144. Form<Void> prefs = new Form<Void>("prefsForm");
  145. List<Language> languages = Arrays.asList(
  146. new Language("English","en"),
  147. new Language("Español", "es"),
  148. new Language("Français", "fr"),
  149. new Language("日本語", "ja"),
  150. new Language("한국말", "ko"),
  151. new Language("Nederlands", "nl"),
  152. new Language("Norsk", "no"),
  153. new Language("Język Polski", "pl"),
  154. new Language("Português", "pt_BR"),
  155. new Language("中文", "zh_CN"));
  156. Locale locale = user.getPreferences().getLocale();
  157. if (locale == null) {
  158. // user has not specified language preference
  159. // try server default preference
  160. String lc = app().settings().getString(Keys.web.forceDefaultLocale, null);
  161. if (StringUtils.isEmpty(lc)) {
  162. // server default language is not configured
  163. // try browser preference
  164. Locale sessionLocale = GitBlitWebSession.get().getLocale();
  165. if (sessionLocale != null) {
  166. locale = sessionLocale;
  167. }
  168. } else {
  169. }
  170. }
  171. Language preferredLanguage = null;
  172. if (locale != null) {
  173. String localeCode = locale.getLanguage();
  174. if (!StringUtils.isEmpty(locale.getCountry())) {
  175. localeCode += "_" + locale.getCountry();
  176. }
  177. for (Language language : languages) {
  178. if (language.code.equals(localeCode)) {
  179. // language_COUNTRY match
  180. preferredLanguage = language;
  181. } else if (preferredLanguage != null && language.code.startsWith(locale.getLanguage())) {
  182. // language match
  183. preferredLanguage = language;
  184. }
  185. }
  186. }
  187. final IModel<String> displayName = Model.of(user.getDisplayName());
  188. final IModel<String> emailAddress = Model.of(user.emailAddress == null ? "" : user.emailAddress);
  189. final IModel<Language> language = Model.of(preferredLanguage);
  190. final IModel<Boolean> emailMeOnMyTicketChanges = Model.of(user.getPreferences().isEmailMeOnMyTicketChanges());
  191. final IModel<Transport> transport = Model.of(user.getPreferences().getTransport());
  192. prefs.add(new TextOption("displayName",
  193. getString("gb.displayName"),
  194. getString("gb.displayNameDescription"),
  195. displayName).setVisible(app().authentication().supportsDisplayNameChanges(user)));
  196. prefs.add(new TextOption("emailAddress",
  197. getString("gb.emailAddress"),
  198. getString("gb.emailAddressDescription"),
  199. emailAddress).setVisible(app().authentication().supportsEmailAddressChanges(user)));
  200. prefs.add(new ChoiceOption<Language>("language",
  201. getString("gb.languagePreference"),
  202. getString("gb.languagePreferenceDescription"),
  203. language,
  204. languages));
  205. prefs.add(new BooleanOption("emailMeOnMyTicketChanges",
  206. getString("gb.emailMeOnMyTicketChanges"),
  207. getString("gb.emailMeOnMyTicketChangesDescription"),
  208. emailMeOnMyTicketChanges).setVisible(app().notifier().isSendingMail()));
  209. List<Transport> availableTransports = new ArrayList<>();
  210. if (app().gitblit().isServingSSH()) {
  211. availableTransports.add(Transport.SSH);
  212. }
  213. if (app().gitblit().isServingHTTP()) {
  214. availableTransports.add(Transport.HTTPS);
  215. availableTransports.add(Transport.HTTP);
  216. }
  217. if (app().gitblit().isServingGIT()) {
  218. availableTransports.add(Transport.GIT);
  219. }
  220. prefs.add(new ChoiceOption<Transport>("transport",
  221. getString("gb.transportPreference"),
  222. getString("gb.transportPreferenceDescription"),
  223. transport,
  224. availableTransports));
  225. prefs.add(new AjaxButton("save") {
  226. private static final long serialVersionUID = 1L;
  227. @Override
  228. protected void onSubmit(AjaxRequestTarget target, Form<?> form) {
  229. UserModel user = GitBlitWebSession.get().getUser();
  230. user.displayName = displayName.getObject();
  231. user.emailAddress = emailAddress.getObject();
  232. Language lang = language.getObject();
  233. if (lang != null) {
  234. user.getPreferences().setLocale(lang.code);
  235. }
  236. user.getPreferences().setEmailMeOnMyTicketChanges(emailMeOnMyTicketChanges.getObject());
  237. user.getPreferences().setTransport(transport.getObject());
  238. try {
  239. app().gitblit().reviseUser(user.username, user);
  240. setRedirect(true);
  241. setResponsePage(UserPage.class, WicketUtils.newUsernameParameter(user.username));
  242. } catch (GitBlitException e) {
  243. // logger.error("Failed to update user " + user.username, e);
  244. // error(getString("gb.failedToUpdateUser"), false);
  245. }
  246. }
  247. });
  248. // add the preferences tab
  249. add(new Fragment("preferencesLink", "preferencesLinkFragment", this).setRenderBodyOnly(true));
  250. Fragment fragment = new Fragment("preferencesTab", "preferencesTabFragment", this);
  251. fragment.add(prefs);
  252. add(fragment.setRenderBodyOnly(true));
  253. }
  254. private void addSshKeys(final UserModel user) {
  255. Fragment keysTab = new Fragment("sshKeysTab", "sshKeysTabFragment", this);
  256. keysTab.add(new SshKeysPanel("sshKeysPanel", user));
  257. // add the SSH keys tab
  258. add(new Fragment("sshKeysLink", "sshKeysLinkFragment", this).setRenderBodyOnly(true));
  259. add(keysTab.setRenderBodyOnly(true));
  260. }
  261. private class Language implements Serializable {
  262. private static final long serialVersionUID = 1L;
  263. final String name;
  264. final String code;
  265. public Language(String name, String code) {
  266. this.name = name;
  267. this.code = code;
  268. }
  269. @Override
  270. public String toString() {
  271. return name + " (" + code +")";
  272. }
  273. }
  274. }