
These bindings can be replaced wholesale with the more idiomatic alternative: aliases. In addition, avoid passing in user to components where it can be injected directly. One of the perceived downsides of dependency injection can be that it can make debugging feel more difficult because it's not immediately clear where the value is coming from, which the explicit variant we previously used does not suffer from. It might also be argued that we also lose out on a seam that could be useful in the future where a component doesn't care about the specific type of user, just that one is passed in. While explicitness is often a virtue, it comes at the cost of increased noise that pervades multiple layers of components. I'd argue this makes the parent components more difficult to understand, given they are littered with unnecessary references to data they themselves do not need. This decreases the noise/ceremony around accessing userPermissions/auth data and restricts access to that data to the child components that actually need to know about it. As to losing a seam, it appears 1) that this isn't currently necessary and 2) we can use an internal computed property should the need arise in the future.
250 lines
6.8 KiB
JavaScript
250 lines
6.8 KiB
JavaScript
import Ember from 'ember';
|
|
import Repo from 'travis/models/repo';
|
|
import Config from 'travis/config/environment';
|
|
|
|
const { service, controller } = Ember.inject;
|
|
const { alias } = Ember.computed;
|
|
|
|
var sortCallback = function(repo1, repo2) {
|
|
// this function could be made simpler, but I think it's clearer this way
|
|
// what're we really trying to achieve
|
|
|
|
var lastBuildId1 = repo1.get('lastBuildId');
|
|
var lastBuildId2 = repo2.get('lastBuildId');
|
|
|
|
if(!lastBuildId1 && !lastBuildId2) {
|
|
// if both repos lack builds, put newer repo first
|
|
return repo1.get('id') > repo2.get('id') ? -1 : 1;
|
|
} else if(lastBuildId1 && !lastBuildId2) {
|
|
// if only repo1 has a build, it goes first
|
|
return -1;
|
|
} else if(lastBuildId2 && !lastBuildId1) {
|
|
// if only repo2 has a build, it goes first
|
|
return 1;
|
|
}
|
|
|
|
var finishedAt1 = repo1.get('lastBuildFinishedAt');
|
|
var finishedAt2 = repo2.get('lastBuildFinishedAt');
|
|
|
|
if(finishedAt1) {
|
|
finishedAt1 = new Date(finishedAt1);
|
|
}
|
|
if(finishedAt2) {
|
|
finishedAt2 = new Date(finishedAt2);
|
|
}
|
|
|
|
if(finishedAt1 && finishedAt2) {
|
|
// if both builds finished, put newer first
|
|
return finishedAt1.getTime() > finishedAt2.getTime() ? -1 : 1;
|
|
} else if(finishedAt1 && !finishedAt2) {
|
|
// if repo1 finished, but repo2 didn't, put repo2 first
|
|
return 1;
|
|
} else if(finishedAt2 && !finishedAt1) {
|
|
// if repo2 finisher, but repo1 didn't, put repo1 first
|
|
return -1;
|
|
} else {
|
|
// none of the builds finished, put newer build first
|
|
return lastBuildId1 > lastBuildId2 ? -1 : 1;
|
|
}
|
|
|
|
throw "should not happen";
|
|
};
|
|
|
|
|
|
|
|
var Controller = Ember.Controller.extend({
|
|
auth: service(),
|
|
ajax: service(),
|
|
updateTimesService: service('updateTimes'),
|
|
|
|
actions: {
|
|
activate: function(name) {
|
|
return this.activate(name);
|
|
},
|
|
showRunningJobs: function() {
|
|
return this.activate('running');
|
|
},
|
|
showMyRepositories: function() {
|
|
if (this.get('tab') === 'running') {
|
|
return this.activate('owned');
|
|
} else {
|
|
return this.transitionToRoute('main.repositories');
|
|
}
|
|
}
|
|
},
|
|
|
|
tabOrIsLoadedDidChange: function() {
|
|
return this.possiblyRedirectToGettingStartedPage();
|
|
}.observes('isLoaded', 'tab', 'repos.length'),
|
|
|
|
possiblyRedirectToGettingStartedPage() {
|
|
return Ember.run.scheduleOnce('routerTransitions', this, function() {
|
|
if (this.get('tab') === 'owned' && this.get('isLoaded') && this.get('repos.length') === 0) {
|
|
return this.container.lookup('router:main').send('redirectToGettingStarted');
|
|
}
|
|
});
|
|
},
|
|
|
|
isLoaded: false,
|
|
repoController: controller('repo'),
|
|
currentUser: alias('auth.currentUser'),
|
|
|
|
selectedRepo: function() {
|
|
return this.get('repoController.repo.content') || this.get('repoController.repo');
|
|
}.property('repoController.repo', 'repoController.repo.content'),
|
|
|
|
startedJobsCount: Ember.computed.alias('runningJobs.length'),
|
|
|
|
allJobsCount: function() {
|
|
return this.get('startedJobsCount') + this.get('queuedJobs.length');
|
|
}.property('startedJobsCount', 'queuedJobs.length'),
|
|
|
|
init() {
|
|
this._super.apply(this, arguments);
|
|
if (!Ember.testing) {
|
|
Visibility.every(this.config.intervals.updateTimes, this.updateTimes.bind(this));
|
|
}
|
|
},
|
|
|
|
runningJobs: function() {
|
|
if(!this.get('config.pro')) { return []; }
|
|
var result;
|
|
|
|
result = this.store.filter('job', {}, function(job) {
|
|
return ['queued', 'started', 'received'].indexOf(job.get('state')) !== -1;
|
|
});
|
|
result.set('isLoaded', false);
|
|
result.then(function() {
|
|
return result.set('isLoaded', true);
|
|
});
|
|
|
|
return result;
|
|
}.property('config.pro'),
|
|
|
|
queuedJobs: function() {
|
|
if(!this.get('config.pro')) { return []; }
|
|
|
|
var result;
|
|
result = this.get('store').filter('job', function(job) {
|
|
return ['created'].indexOf(job.get('state')) !== -1;
|
|
});
|
|
result.set('isLoaded', false);
|
|
result.then(function() {
|
|
result.set('isLoaded', true);
|
|
});
|
|
|
|
return result;
|
|
}.property('config.pro'),
|
|
|
|
recentRepos: function() {
|
|
return [];
|
|
}.property(),
|
|
|
|
updateTimes() {
|
|
let records = this.get('repos');
|
|
|
|
if(Config.useV3API) {
|
|
let callback = (record) => { return record.get('lastBuild'); };
|
|
records = records.filter(callback).map(callback);
|
|
}
|
|
this.get('updateTimesService').push(records);
|
|
},
|
|
|
|
activate(tab, params) {
|
|
this.set('sortProperties', ['sortOrder']);
|
|
this.set('tab', tab);
|
|
return this[("view_" + tab).camelize()](params);
|
|
},
|
|
|
|
viewOwned() {
|
|
var repos, user;
|
|
|
|
if (repos = this.get('ownedRepos')) {
|
|
return this.set('_repos', repos);
|
|
} else if (!this.get('fetchingOwnedRepos')) {
|
|
this.set('fetchingOwnedRepos', true);
|
|
this.set('isLoaded', false);
|
|
|
|
if (user = this.get('currentUser')) {
|
|
let callback = (reposRecordArray) => {
|
|
this.set('isLoaded', true);
|
|
this.set('_repos', reposRecordArray);
|
|
this.set('ownedRepos', reposRecordArray);
|
|
this.set('fetchingOwnedRepos', false);
|
|
return reposRecordArray;
|
|
};
|
|
|
|
if(Config.useV3API) {
|
|
user.get('_rawPermissions').then( (data) => {
|
|
Repo.accessibleBy(this.store, data.pull).then(callback);
|
|
});
|
|
} else {
|
|
let login = user.get('login');
|
|
Repo.accessibleBy(this.store, login).then(callback);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
viewRunning() {},
|
|
|
|
viewSearch(phrase) {
|
|
this.set('search', phrase);
|
|
this.set('isLoaded', false);
|
|
Repo.search(this.store, this.get('ajax'), phrase).then( (reposRecordArray) => {
|
|
this.set('isLoaded', true);
|
|
this.set('_repos', reposRecordArray);
|
|
});
|
|
},
|
|
|
|
searchObserver: function() {
|
|
var search;
|
|
search = this.get('search');
|
|
if (search) {
|
|
return this.searchFor(search);
|
|
}
|
|
}.observes('search'),
|
|
|
|
searchFor(phrase) {
|
|
if (this.searchLater) {
|
|
Ember.run.cancel(this.searchLater);
|
|
}
|
|
this.searchLater = Ember.run.later(this, (function() {
|
|
this.transitionToRoute('main.search', phrase.replace(/\//g, '%2F'));
|
|
}), 500);
|
|
},
|
|
|
|
noReposMessage: function() {
|
|
var tab;
|
|
tab = this.get('tab');
|
|
if (tab === 'owned') {
|
|
return 'You don\'t have any repos set up on Travis CI';
|
|
} else if (tab === 'recent') {
|
|
return 'Repositories could not be loaded';
|
|
} else {
|
|
return 'Could not find any repos';
|
|
}
|
|
}.property('tab'),
|
|
|
|
showRunningJobs: function() {
|
|
return this.get('tab') === 'running';
|
|
}.property('tab'),
|
|
|
|
repos: function() {
|
|
var repos = this.get('_repos');
|
|
|
|
if(repos && repos.toArray) {
|
|
repos = repos.toArray();
|
|
}
|
|
|
|
if(repos && repos.sort) {
|
|
return repos.sort(sortCallback);
|
|
} else {
|
|
return [];
|
|
}
|
|
}.property('_repos.[]', '_repos.@each.lastBuildFinishedAt',
|
|
'_repos.@each.lastBuildId')
|
|
});
|
|
|
|
export default Controller;
|