function getFirstTextLine(commentString) { if (!commentString) { return; } var lines = commentString.split('\n'); var MARKDOWN_CHARS = { '#': true, '-': true, '>': true, '`': true }; for (var i = 0; i < lines.length; ++i) { // Skip code lines. if (lines[i].indexOf(' ') == 0) { continue; } // Skip empty lines. if ($.trim(lines[i]).length == 0) { continue; } // Skip control lines. if (MARKDOWN_CHARS[$.trim(lines[i])[0]]) { continue; } return getMarkedDown(lines[i]); } return ''; } function getMarkedDown(string) { return Markdown.getSanitizingConverter().makeHtml(string || ''); } function HeaderCtrl($scope, $location, UserService, Restangular) { $scope.$watch( function () { return UserService.currentUser(); }, function (currentUser) { $scope.user = currentUser; }, true); $scope.signout = function() { var signoutPost = Restangular.one('signout'); signoutPost.customPOST().then(function() { UserService.load(); $location.path('/'); }); } $scope.$on('$includeContentLoaded', function() { // THIS IS BAD, MOVE THIS TO A DIRECTIVE $('#repoSearch').typeahead({ name: 'repositories', remote: { url: '/api/find/repository?query=%QUERY', filter: function(data) { var datums = []; for (var i = 0; i < data.repositories.length; ++i) { var repo = data.repositories[i]; datums.push({ 'value': repo.name, 'tokens': [repo.name, repo.namespace], 'repo': repo }); } return datums; } }, template: function (datum) { template = '
'; template += '' template += '' + datum.repo.namespace +'/' + datum.repo.name + '' if (datum.repo.description) { template += '' + getFirstTextLine(datum.repo.description) + '' } template += '
' return template; }, }); $('#repoSearch').on('typeahead:selected', function (e, datum) { $('#repoSearch').typeahead('setQuery', ''); document.location = '/repository/' + datum.repo.namespace + '/' + datum.repo.name }); }); } function SigninCtrl($scope, $location, $timeout, Restangular, KeyService, UserService) { $scope.githubClientId = KeyService.githubClientId; var appendMixpanelId = function() { if (mixpanel.get_distinct_id !== undefined) { $scope.mixpanelDistinctIdClause = "&state=" + mixpanel.get_distinct_id(); } else { // Mixpanel not yet loaded, try again later $timeout(appendMixpanelId, 200); } }; appendMixpanelId(); $scope.signin = function() { var signinPost = Restangular.one('signin'); signinPost.customPOST($scope.user).then(function() { $scope.needsEmailVerification = false; $scope.invalidCredentials = false; // Redirect to the landing page UserService.load(); $location.path('/'); }, function(result) { $scope.needsEmailVerification = result.data.needsEmailVerification; $scope.invalidCredentials = result.data.invalidCredentials; }); }; $scope.sendRecovery = function() { var signinPost = Restangular.one('recovery'); signinPost.customPOST($scope.recovery).then(function() { $scope.invalidEmail = false; $scope.sent = true; }, function(result) { $scope.invalidEmail = true; $scope.sent = false; }); }; }; function PlansCtrl($scope, UserService, PlanService) { $scope.plans = PlanService.planList(); $scope.$watch( function () { return UserService.currentUser(); }, function (currentUser) { $scope.user = currentUser; }, true); $scope.buyNow = function(plan) { if ($scope.user && !$scope.user.anonymous) { document.location = '/user?plan=' + plan; } else { $('#signinModal').modal({}); } }; $scope.status = 'ready'; } function GuideCtrl($scope) { $scope.status = 'ready'; } function RepoListCtrl($scope, Restangular, UserService) { $scope.$watch( function () { return UserService.currentUser(); }, function (currentUser) { $scope.user = currentUser; }, true); $scope.getCommentFirstLine = function(commentString) { return getMarkedDown(getFirstTextLine(commentString)); }; $scope.getMarkedDown = function(string) { if (!string) { return ''; } return getMarkedDown(string); }; $scope.loading = true; $scope.public_repositories = null; $scope.private_repositories = null; // Load the list of personal repositories. var repositoryPrivateFetch = Restangular.all('repository/'); repositoryPrivateFetch.getList({'public': false, 'sort': true}).then(function(resp) { $scope.private_repositories = resp.repositories; $scope.loading = !($scope.public_repositories && $scope.private_repositories); }); // Load the list of public repositories. var options = {'public': true, 'private': false, 'sort': true, 'limit': 10}; var repositoryPublicFetch = Restangular.all('repository/'); repositoryPublicFetch.getList(options).then(function(resp) { $scope.public_repositories = resp.repositories; $scope.loading = !($scope.public_repositories && $scope.private_repositories); }); } function LandingCtrl($scope, $timeout, Restangular, UserService, KeyService) { $('.form-signup').popover(); $scope.$watch( function () { return UserService.currentUser(); }, function (currentUser) { if (!currentUser.anonymous) { $scope.loadMyRepos(); } $scope.user = currentUser; }, true); angulartics.waitForVendorApi(mixpanel, 500, function(loadedMixpanel) { var mixpanelId = loadedMixpanel.get_distinct_id(); $scope.github_state_clause = '&state=' + mixpanelId; }); $scope.githubClientId = KeyService.githubClientId; $scope.awaitingConfirmation = false; $scope.registering = false; $scope.getCommentFirstLine = function(commentString) { return getMarkedDown(getFirstTextLine(commentString)); }; $scope.browseRepos = function() { document.location = '/repository/'; }; $scope.register = function() { $('.form-signup').popover('hide'); $scope.registering = true; var newUserPost = Restangular.one('user/'); newUserPost.customPOST($scope.newUser).then(function() { $scope.awaitingConfirmation = true; $scope.registering = false; mixpanel.alias($scope.newUser.username); }, function(result) { $scope.registering = false; $scope.registerError = result.data.message; $timeout(function() { $('.form-signup').popover('show'); }); }); }; $scope.loadMyRepos = function() { $scope.loadingmyrepos = true; // Load the list of repositories. var params = { 'limit': 5, 'public': false, 'sort': true }; var repositoryFetch = Restangular.all('repository/'); repositoryFetch.getList(params).then(function(resp) { $scope.myrepos = resp.repositories; $scope.loadingmyrepos = false; }); }; $scope.status = 'ready'; browserchrome.update(); } function RepoCtrl($scope, Restangular, $routeParams, $rootScope, $location) { $rootScope.title = 'Loading...'; // Watch for changes to the tag parameter. $scope.$on('$routeUpdate', function(){ $scope.setTag($location.search().tag, false); }); $scope.editDescription = function() { if (!$scope.repo.can_write) { return; } if (!$scope.markdownDescriptionEditor) { var converter = Markdown.getSanitizingConverter(); var editor = new Markdown.Editor(converter, '-description'); editor.run(); $scope.markdownDescriptionEditor = editor; } $('#wmd-input-description')[0].value = $scope.repo.description; $('#editModal').modal({}); }; $scope.saveDescription = function() { $('#editModal').modal('hide'); $scope.repo.description = $('#wmd-input-description')[0].value; $scope.repo.put(); }; $scope.parseDate = function(dateString) { return Date.parse(dateString); }; $scope.getCommentFirstLine = function(commentString) { return getMarkedDown(getFirstTextLine(commentString)); }; $scope.getTimeSince = function(createdTime) { return moment($scope.parseDate(createdTime)).fromNow(); }; $scope.getMarkedDown = function(string) { if (!string) { return ''; } return getMarkedDown(string); }; var getDefaultTag = function() { if ($scope.repo === undefined) { return undefined; } else if ($scope.repo.tags.hasOwnProperty('latest')) { return $scope.repo.tags['latest']; } else { for (key in $scope.repo.tags) { return $scope.repo.tags[key]; } } }; var listImages = function() { if ($scope.imageHistory) { return; } var imageFetch = Restangular.one('repository/' + namespace + '/' + name + '/image/'); imageFetch.get().then(function(resp) { $scope.imageHistory = resp.images; $scope.tree = new ImageHistoryTree(namespace, name, resp.images, $scope.getCommentFirstLine, $scope.getTimeSince); $scope.tree.draw('image-history-container'); // If we already have a tag, use it if ($scope.currentTag) { $scope.tree.setTag($scope.currentTag.name); } $($scope.tree).bind('tagChanged', function(e) { $scope.$apply(function() { $scope.setTag(e.tag, true); }); }); $($scope.tree).bind('imageChanged', function(e) { $scope.$apply(function() { $scope.setImage(e.image); }); }); }); }; $scope.setImage = function(image) { $scope.currentImage = image; if ($scope.tree) { $scope.tree.setImage($scope.currentImage.id); } }; $scope.setTag = function(tagName, opt_updateURL) { var repo = $scope.repo; var proposedTag = repo.tags[tagName]; if (!proposedTag) { // We must find a good default for (tagName in repo.tags) { if (!proposedTag || tagName == 'latest') { proposedTag = repo.tags[tagName]; } } } if (proposedTag) { $scope.currentTag = proposedTag; $scope.currentImage = $scope.currentTag.image; if ($scope.tree) { $scope.tree.setTag($scope.currentTag.name); } if (opt_updateURL) { $location.search('tag', $scope.currentTag.name); } } }; $scope.getTagCount = function(repo) { if (!repo) { return 0; } var count = 0; for (var tag in repo.tags) { ++count; } return count; }; var namespace = $routeParams.namespace; var name = $routeParams.name; $scope.loading = true; // Fetch the repo. var repositoryFetch = Restangular.one('repository/' + namespace + '/' + name); repositoryFetch.get().then(function(repo) { $rootScope.title = namespace + '/' + name; $scope.repo = repo; $scope.setTag($routeParams.tag); var clip = new ZeroClipboard($('#copyClipboard'), { 'moviePath': 'static/lib/ZeroClipboard.swf' }); clip.on('complete', function() { // Resets the animation. var elem = $('#clipboardCopied')[0]; elem.style.display = 'none'; // Show the notification. setTimeout(function() { elem.style.display = 'block'; }, 1); }); $scope.loading = false; }, function() { $scope.repo = null; $scope.loading = false; $rootScope.title = 'Unknown Repository'; }); // Fetch the image history. listImages(); } function RepoAdminCtrl($scope, Restangular, $routeParams, $rootScope) { var namespace = $routeParams.namespace; var name = $routeParams.name; $scope.$on('$viewContentLoaded', function() { // THIS IS BAD, MOVE THIS TO A DIRECTIVE $('#userSearch').typeahead({ name: 'users', remote: { url: '/api/users/%QUERY', filter: function(data) { var datums = []; for (var i = 0; i < data.users.length; ++i) { var user = data.users[i]; datums.push({ 'value': user, 'tokens': [user], 'username': user }); } return datums; } }, template: function (datum) { template = '
'; template += '' template += '' + datum.username + '' template += '
' return template; }, }); $('#userSearch').on('typeahead:selected', function(e, datum) { $('#userSearch').typeahead('setQuery', ''); $scope.addNewPermission(datum.username); }); }); $scope.addNewPermission = function(username) { // Don't allow duplicates. if ($scope.permissions[username]) { return; } // Need the $scope.apply for both the permission stuff to change and for // the XHR call to be made. $scope.$apply(function() { $scope.addRole(username, 'read') }); }; $scope.deleteRole = function(username) { var permissionDelete = Restangular.one('repository/' + namespace + '/' + name + '/permissions/' + username); permissionDelete.customDELETE().then(function() { delete $scope.permissions[username]; }, function(result) { if (result.status == 409) { $('#onlyadminModal').modal({}); } else { $('#cannotchangeModal').modal({}); } }); }; $scope.addRole = function(username, role) { var permission = { 'role': role }; var permissionPost = Restangular.one('repository/' + namespace + '/' + name + '/permissions/' + username); permissionPost.customPOST(permission).then(function() { $scope.permissions[username] = permission; $scope.permissions = $scope.permissions; }, function(result) { $('#cannotchangeModal').modal({}); }); }; $scope.setRole = function(username, role) { var permission = $scope.permissions[username]; var currentRole = permission.role; permission.role = role; var permissionPut = Restangular.one('repository/' + namespace + '/' + name + '/permissions/' + username); permissionPut.customPUT(permission).then(function() {}, function(result) { if (result.status == 409) { permission.role = currentRole; $('#onlyadminModal').modal({}); } else { $('#cannotchangeModal').modal({}); } }); }; $scope.askChangeAccess = function(newAccess) { $('#make' + newAccess + 'Modal').modal({}); }; $scope.changeAccess = function(newAccess) { $('#make' + newAccess + 'Modal').modal('hide'); var visibility = { 'visibility': newAccess }; var visibilityPost = Restangular.one('repository/' + namespace + '/' + name + '/changevisibility'); visibilityPost.customPOST(visibility).then(function() { $scope.repo.is_public = newAccess == 'public'; }, function() { $('#cannotchangeModal').modal({}); }); }; $scope.askDelete = function() { $('#confirmdeleteModal').modal({}); }; $scope.deleteRepo = function() { $('#confirmdeleteModal').modal('hide'); var deleteAction = Restangular.one('repository/' + namespace + '/' + name); deleteAction.customDELETE().then(function() { $scope.repo = null; setTimeout(function() { document.location = '/repository/'; }, 1000); }, function() { $('#cannotchangeModal').modal({}); }); }; $scope.loading = true; // Fetch the repository information. var repositoryFetch = Restangular.one('repository/' + namespace + '/' + name); repositoryFetch.get().then(function(repo) { $scope.repo = repo; $scope.loading = !($scope.permissions && $scope.repo); }, function() { $scope.permissions = null; $rootScope.title = 'Unknown Repository'; $scope.loading = false; }); // Fetch the permissions. var permissionsFetch = Restangular.one('repository/' + namespace + '/' + name + '/permissions'); permissionsFetch.get().then(function(resp) { $rootScope.title = 'Settings - ' + namespace + '/' + name; $scope.permissions = resp.permissions; $scope.loading = !($scope.permissions && $scope.repo); }, function() { $scope.permissions = null; $rootScope.title = 'Unknown Repository'; $scope.loading = false; }); } function UserAdminCtrl($scope, $timeout, Restangular, PlanService, UserService, KeyService, $routeParams) { $scope.plans = PlanService.planList(); $scope.$watch(function () { return UserService.currentUser(); }, function (currentUser) { $scope.askForPassword = currentUser.askForPassword; }, true); var subscribedToPlan = function(sub) { $scope.subscription = sub; $scope.subscribedPlan = PlanService.getPlan(sub.plan); $scope.planUsagePercent = sub.usedPrivateRepos * 100 / $scope.subscribedPlan.privateRepos; if (sub.usedPrivateRepos > $scope.subscribedPlan.privateRepos) { $scope.errorMessage = 'You are using more private repositories than your plan allows, please upgrate your subscription to avoid disruptions in your service.'; } $scope.planLoading = false; $scope.planChanging = false; mixpanel.people.set({ 'plan': sub.plan }); } $scope.planLoading = true; var getSubscription = Restangular.one('user/plan'); getSubscription.get().then(subscribedToPlan, function() { // User has no subscription $scope.planLoading = false; }); $scope.planChanging = false; $scope.subscribe = function(planId) { var submitToken = function(token) { $scope.$apply(function() { mixpanel.track('plan_subscribe'); $scope.planChanging = true; $scope.errorMessage = undefined; var subscriptionDetails = { token: token.id, plan: planId, }; var createSubscriptionRequest = Restangular.one('user/plan'); createSubscriptionRequest.customPUT(subscriptionDetails).then(subscribedToPlan, function() { // Failure $scope.errorMessage = 'Unable to subscribe.'; }); }); }; var planDetails = PlanService.getPlan(planId) StripeCheckout.open({ key: KeyService.stripePublishableKey, address: false, // TODO change to true amount: planDetails.price, currency: 'usd', name: 'Quay ' + planDetails.title + ' Subscription', description: 'Up to ' + planDetails.privateRepos + ' private repositories', panelLabel: 'Subscribe', token: submitToken }); }; $scope.changeSubscription = function(planId) { $scope.planChanging = true; $scope.errorMessage = undefined; var subscriptionDetails = { plan: planId, }; var changeSubscriptionRequest = Restangular.one('user/plan'); changeSubscriptionRequest.customPUT(subscriptionDetails).then(subscribedToPlan, function() { // Failure $scope.errorMessage = 'Unable to change subscription.'; $scope.planChanging = false; }); }; $scope.cancelSubscription = function() { $scope.changeSubscription('free'); }; // Show the subscribe dialog if a plan was requested. var requested = $routeParams['plan'] if (requested !== undefined && requested !== 'free') { if (PlanService.getPlan(requested) !== undefined) { $scope.subscribe(requested); } } $scope.updatingUser = false; $scope.changePasswordSuccess = false; $('.form-change-pw').popover(); $scope.changePassword = function() { $('.form-change-pw').popover('hide'); $scope.updatingUser = true; $scope.changePasswordSuccess = false; var changePasswordPost = Restangular.one('user/'); changePasswordPost.customPUT($scope.user).then(function() { $scope.updatingUser = false; $scope.changePasswordSuccess = true; // Reset the form $scope.user.password = ''; $scope.user.repeatPassword = ''; $scope.changePasswordForm.$setPristine(); UserService.load(); }, function(result) { $scope.updatingUser = false; $scope.changePasswordError = result.data.message; $timeout(function() { $('.form-change-pw').popover('show'); }); }); }; }