FEATURE: manage Permalinks
This commit is contained in:
parent
7f43653cad
commit
dc90c396f2
|
@ -0,0 +1,56 @@
|
||||||
|
export default Ember.Component.extend({
|
||||||
|
classNames: ['permalink-form'],
|
||||||
|
formSubmitted: false,
|
||||||
|
permalinkType: 'topic_id',
|
||||||
|
|
||||||
|
permalinkTypes: function() {
|
||||||
|
return [
|
||||||
|
{id: 'topic_id', name: I18n.t('admin.permalink.topic_id')},
|
||||||
|
{id: 'post_id', name: I18n.t('admin.permalink.post_id')},
|
||||||
|
{id: 'category_id', name: I18n.t('admin.permalink.category_id')},
|
||||||
|
{id: 'external_url', name: I18n.t('admin.permalink.external_url')}
|
||||||
|
];
|
||||||
|
}.property(),
|
||||||
|
|
||||||
|
permalinkTypePlaceholder: function() {
|
||||||
|
return 'admin.permalink.' + this.get('permalinkType');
|
||||||
|
}.property('permalinkType'),
|
||||||
|
|
||||||
|
actions: {
|
||||||
|
submit: function() {
|
||||||
|
if (!this.get('formSubmitted')) {
|
||||||
|
const self = this;
|
||||||
|
self.set('formSubmitted', true);
|
||||||
|
const permalink = Discourse.Permalink.create({url: self.get('url'), permalink_type: self.get('permalinkType'), permalink_type_value: self.get('permalink_type_value')});
|
||||||
|
permalink.save().then(function(result) {
|
||||||
|
self.set('url', '');
|
||||||
|
self.set('permalink_type_value', '');
|
||||||
|
self.set('formSubmitted', false);
|
||||||
|
self.sendAction('action', Discourse.Permalink.create(result.permalink));
|
||||||
|
Em.run.schedule('afterRender', function() { self.$('.permalink-url').focus(); });
|
||||||
|
}, function(e) {
|
||||||
|
self.set('formSubmitted', false);
|
||||||
|
let error;
|
||||||
|
if (e.responseJSON && e.responseJSON.errors) {
|
||||||
|
error = I18n.t("generic_error_with_reason", {error: e.responseJSON.errors.join('. ')});
|
||||||
|
} else {
|
||||||
|
error = I18n.t("generic_error");
|
||||||
|
}
|
||||||
|
bootbox.alert(error, function() { self.$('.permalink-url').focus(); });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
didInsertElement: function() {
|
||||||
|
var self = this;
|
||||||
|
self._super();
|
||||||
|
Em.run.schedule('afterRender', function() {
|
||||||
|
self.$('.external-url').keydown(function(e) {
|
||||||
|
if (e.keyCode === 13) { // enter key
|
||||||
|
self.send('submit');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
|
@ -0,0 +1,36 @@
|
||||||
|
export default Ember.ArrayController.extend({
|
||||||
|
loading: false,
|
||||||
|
filter: null,
|
||||||
|
|
||||||
|
show: Discourse.debounce(function() {
|
||||||
|
var self = this;
|
||||||
|
self.set('loading', true);
|
||||||
|
Discourse.Permalink.findAll(self.get("filter")).then(function(result) {
|
||||||
|
self.set('model', result);
|
||||||
|
self.set('loading', false);
|
||||||
|
});
|
||||||
|
}, 250).observes("filter"),
|
||||||
|
|
||||||
|
actions: {
|
||||||
|
recordAdded(arg) {
|
||||||
|
this.get("model").unshiftObject(arg);
|
||||||
|
},
|
||||||
|
|
||||||
|
destroy: function(record) {
|
||||||
|
const self = this;
|
||||||
|
return bootbox.confirm(I18n.t("admin.permalink.delete_confirm"), I18n.t("no_value"), I18n.t("yes_value"), function(result) {
|
||||||
|
if (result) {
|
||||||
|
record.destroy().then(function(deleted) {
|
||||||
|
if (deleted) {
|
||||||
|
self.removeObject(record);
|
||||||
|
} else {
|
||||||
|
bootbox.alert(I18n.t("generic_error"));
|
||||||
|
}
|
||||||
|
}, function(){
|
||||||
|
bootbox.alert(I18n.t("generic_error"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
|
@ -0,0 +1,22 @@
|
||||||
|
const Permalink = Discourse.Model.extend({
|
||||||
|
save: function() {
|
||||||
|
return Discourse.ajax("/admin/permalinks.json", {
|
||||||
|
type: 'POST',
|
||||||
|
data: {url: this.get('url'), permalink_type: this.get('permalink_type'), permalink_type_value: this.get('permalink_type_value')}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
destroy: function() {
|
||||||
|
return Discourse.ajax("/admin/permalinks/" + this.get('id') + ".json", {type: 'DELETE'});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Permalink.reopenClass({
|
||||||
|
findAll: function(filter) {
|
||||||
|
return Discourse.ajax("/admin/permalinks.json", { data: { filter: filter } }).then(function(permalinks) {
|
||||||
|
return permalinks.map(p => Discourse.Permalink.create(p));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
export default Permalink;
|
|
@ -0,0 +1,9 @@
|
||||||
|
export default Discourse.Route.extend({
|
||||||
|
model() {
|
||||||
|
return Discourse.Permalink.findAll();
|
||||||
|
},
|
||||||
|
|
||||||
|
setupController(controller, model) {
|
||||||
|
controller.set('model', model);
|
||||||
|
}
|
||||||
|
});
|
|
@ -22,6 +22,7 @@ export default {
|
||||||
});
|
});
|
||||||
this.resource('adminUserFields', { path: '/user_fields' });
|
this.resource('adminUserFields', { path: '/user_fields' });
|
||||||
this.resource('adminEmojis', { path: '/emojis' });
|
this.resource('adminEmojis', { path: '/emojis' });
|
||||||
|
this.resource('adminPermalinks', { path: '/permalinks' });
|
||||||
});
|
});
|
||||||
this.route('api');
|
this.route('api');
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
<b>{{i18n 'admin.permalink.form.label'}}</b>
|
||||||
|
{{text-field value=url disabled=formSubmitted class="permalink-url" placeholderKey="admin.permalink.url" autocorrect="off" autocapitalize="off"}}
|
||||||
|
{{combo-box content=permalinkTypes value=permalinkType}}
|
||||||
|
{{text-field value=permalink_type_value disabled=formSubmitted class="external-url" placeholderKey=permalinkTypePlaceholder autocorrect="off" autocapitalize="off"}}
|
||||||
|
<button class="btn" {{action "submit" target="view"}} {{bind-attr disabled="formSubmitted"}}>{{i18n 'admin.permalink.form.add'}}</button>
|
|
@ -4,6 +4,7 @@
|
||||||
{{nav-item route='adminSiteText' label='admin.site_text.title'}}
|
{{nav-item route='adminSiteText' label='admin.site_text.title'}}
|
||||||
{{nav-item route='adminUserFields' label='admin.user_fields.title'}}
|
{{nav-item route='adminUserFields' label='admin.user_fields.title'}}
|
||||||
{{nav-item route='adminEmojis' label='admin.emoji.title'}}
|
{{nav-item route='adminEmojis' label='admin.emoji.title'}}
|
||||||
|
{{nav-item route='adminPermalinks' label='admin.permalink.title'}}
|
||||||
{{/admin-nav}}
|
{{/admin-nav}}
|
||||||
|
|
||||||
<div class="admin-container">
|
<div class="admin-container">
|
||||||
|
|
|
@ -0,0 +1,25 @@
|
||||||
|
<div class="permalink-title"><h2>{{i18n 'admin.permalink.title'}}</h2></div>
|
||||||
|
<div class="pull-right">
|
||||||
|
{{text-field value=filter class="url-input" placeholderKey="admin.permalink.form.filter" autocorrect="off" autocapitalize="off"}}
|
||||||
|
</div>
|
||||||
|
{{permalink-form action="recordAdded"}}
|
||||||
|
<br/>
|
||||||
|
|
||||||
|
{{#conditional-loading-spinner condition=loading}}
|
||||||
|
{{#if model.length}}
|
||||||
|
<div class='table admin-logs-table permalinks'>
|
||||||
|
<div class="heading-container">
|
||||||
|
<div class="col heading first url">{{i18n 'admin.permalink.url'}}</div>
|
||||||
|
<div class="col heading topic_id">{{i18n 'admin.permalink.topic_id'}}</div>
|
||||||
|
<div class="col heading post_id">{{i18n 'admin.permalink.post_id'}}</div>
|
||||||
|
<div class="col heading category_id">{{i18n 'admin.permalink.category_id'}}</div>
|
||||||
|
<div class="col heading external_url">{{i18n 'admin.permalink.external_url'}}</div>
|
||||||
|
<div class="col heading actions"></div>
|
||||||
|
<div class="clearfix"></div>
|
||||||
|
</div>
|
||||||
|
{{view 'permalinks-list' content=controller}}
|
||||||
|
</div>
|
||||||
|
{{else}}
|
||||||
|
{{i18n 'search.no_results'}}
|
||||||
|
{{/if}}
|
||||||
|
{{/conditional-loading-spinner}}
|
|
@ -0,0 +1,7 @@
|
||||||
|
<div class="col first url">{{url}}</div>
|
||||||
|
<div class="col topic_id">{{topic_id}}</div>
|
||||||
|
<div class="col post_id">{{post_id}}</div>
|
||||||
|
<div class="col category_id">{{category_id}}</div>
|
||||||
|
<div class="col external_url">{{external_url}}</div>
|
||||||
|
<div class="col action"><button class="btn btn-danger" {{action "destroy" this}}><i class="fa fa-trash-o"></i></button></div>
|
||||||
|
<div class="clearfix"></div>
|
|
@ -0,0 +1,8 @@
|
||||||
|
import ListView from 'ember-addons/list-view';
|
||||||
|
import ListItemView from 'ember-addons/list-item-view';
|
||||||
|
|
||||||
|
export default ListView.extend({
|
||||||
|
height: 700,
|
||||||
|
rowHeight: 32,
|
||||||
|
itemViewClass: ListItemView.extend({templateName: "admin/templates/permalinks_list_item"})
|
||||||
|
});
|
|
@ -1199,7 +1199,7 @@ table.api-keys {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.staff-actions, .screened-emails, .screened-urls, .screened-ip-addresses {
|
.staff-actions, .screened-emails, .screened-urls, .screened-ip-addresses, .permalinks {
|
||||||
|
|
||||||
border-bottom: dotted 1px scale-color($primary, $lightness: 75%);
|
border-bottom: dotted 1px scale-color($primary, $lightness: 75%);
|
||||||
|
|
||||||
|
@ -1469,3 +1469,19 @@ table#user-badges {
|
||||||
width: 90%;
|
width: 90%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Permalinks
|
||||||
|
|
||||||
|
.permalinks {
|
||||||
|
.url, .external_url {
|
||||||
|
width: 300px;
|
||||||
|
}
|
||||||
|
.action, .topic_id, .post_id, .category_id {
|
||||||
|
text-align: center;
|
||||||
|
width: 9.9099%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.permalink-title {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
class Admin::PermalinksController < Admin::AdminController
|
||||||
|
|
||||||
|
before_filter :fetch_permalink, only: [:destroy]
|
||||||
|
|
||||||
|
def index
|
||||||
|
filter = params[:filter]
|
||||||
|
|
||||||
|
permalinks = Permalink
|
||||||
|
permalinks = permalinks.where('url ILIKE :filter OR external_url ILIKE :filter', filter: "%#{params[:filter]}%") if filter.present?
|
||||||
|
permalinks = permalinks.limit(100).order('created_at desc').to_a
|
||||||
|
|
||||||
|
render_serialized(permalinks, PermalinkSerializer)
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
params.require(:url)
|
||||||
|
params.require(:permalink_type)
|
||||||
|
params.require(:permalink_type_value)
|
||||||
|
|
||||||
|
permalink = Permalink.new(:url => params[:url], params[:permalink_type] => params[:permalink_type_value])
|
||||||
|
if permalink.save
|
||||||
|
render_serialized(permalink, PermalinkSerializer)
|
||||||
|
else
|
||||||
|
render_json_error(permalink)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@permalink.destroy
|
||||||
|
render json: success_json
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def fetch_permalink
|
||||||
|
@permalink = Permalink.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -0,0 +1,3 @@
|
||||||
|
class PermalinkSerializer < ApplicationSerializer
|
||||||
|
attributes :id, :url, :topic_id, :post_id, :category_id, :external_url
|
||||||
|
end
|
|
@ -2443,6 +2443,19 @@ en:
|
||||||
image: "Image"
|
image: "Image"
|
||||||
delete_confirm: "Are you sure you want to delete the :%{name}: emoji?"
|
delete_confirm: "Are you sure you want to delete the :%{name}: emoji?"
|
||||||
|
|
||||||
|
permalink:
|
||||||
|
title: "Permalinks"
|
||||||
|
url: "URL"
|
||||||
|
topic_id: "Topic ID"
|
||||||
|
post_id: "Post ID"
|
||||||
|
category_id: "Category ID"
|
||||||
|
external_url: "External URL"
|
||||||
|
delete_confirm: Are you sure you want to delete this permalink?
|
||||||
|
form:
|
||||||
|
label: "New:"
|
||||||
|
add: "Add"
|
||||||
|
filter: "Search (URL or External URL)"
|
||||||
|
|
||||||
lightbox:
|
lightbox:
|
||||||
download: "download"
|
download: "download"
|
||||||
|
|
||||||
|
|
|
@ -133,6 +133,7 @@ Discourse::Application.routes.draw do
|
||||||
get "customize" => "color_schemes#index", constraints: AdminConstraint.new
|
get "customize" => "color_schemes#index", constraints: AdminConstraint.new
|
||||||
get "customize/css_html" => "site_customizations#index", constraints: AdminConstraint.new
|
get "customize/css_html" => "site_customizations#index", constraints: AdminConstraint.new
|
||||||
get "customize/colors" => "color_schemes#index", constraints: AdminConstraint.new
|
get "customize/colors" => "color_schemes#index", constraints: AdminConstraint.new
|
||||||
|
get "customize/permalinks" => "permalinks#index", constraints: AdminConstraint.new
|
||||||
get "flags" => "flags#index"
|
get "flags" => "flags#index"
|
||||||
get "flags/:filter" => "flags#index"
|
get "flags/:filter" => "flags#index"
|
||||||
post "flags/agree/:id" => "flags#agree"
|
post "flags/agree/:id" => "flags#agree"
|
||||||
|
@ -148,6 +149,8 @@ Discourse::Application.routes.draw do
|
||||||
|
|
||||||
resources :color_schemes, constraints: AdminConstraint.new
|
resources :color_schemes, constraints: AdminConstraint.new
|
||||||
|
|
||||||
|
resources :permalinks, constraints: AdminConstraint.new
|
||||||
|
|
||||||
get "version_check" => "versions#show"
|
get "version_check" => "versions#show"
|
||||||
|
|
||||||
resources :dashboard, only: [:index] do
|
resources :dashboard, only: [:index] do
|
||||||
|
|
|
@ -0,0 +1,51 @@
|
||||||
|
require 'spec_helper'
|
||||||
|
|
||||||
|
describe Admin::PermalinksController do
|
||||||
|
|
||||||
|
it "is a subclass of AdminController" do
|
||||||
|
expect(Admin::PermalinksController < Admin::AdminController).to eq(true)
|
||||||
|
end
|
||||||
|
|
||||||
|
let!(:user) { log_in(:admin) }
|
||||||
|
|
||||||
|
describe 'index' do
|
||||||
|
it 'filters url' do
|
||||||
|
Fabricate(:permalink, url: "/forum/23")
|
||||||
|
Fabricate(:permalink, url: "/forum/98")
|
||||||
|
Fabricate(:permalink, url: "/discuss/topic/45")
|
||||||
|
Fabricate(:permalink, url: "/discuss/topic/76")
|
||||||
|
|
||||||
|
xhr :get, :index, filter: "topic"
|
||||||
|
|
||||||
|
expect(response).to be_success
|
||||||
|
result = JSON.parse(response.body)
|
||||||
|
expect(result.length).to eq(2)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'filters external url' do
|
||||||
|
Fabricate(:permalink, external_url: "http://google.com")
|
||||||
|
Fabricate(:permalink, external_url: "http://wikipedia.org")
|
||||||
|
Fabricate(:permalink, external_url: "http://www.discourse.org")
|
||||||
|
Fabricate(:permalink, external_url: "http://try.discourse.org")
|
||||||
|
|
||||||
|
xhr :get, :index, filter: "discourse"
|
||||||
|
|
||||||
|
expect(response).to be_success
|
||||||
|
result = JSON.parse(response.body)
|
||||||
|
expect(result.length).to eq(2)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'filters url and external url both' do
|
||||||
|
Fabricate(:permalink, url: "/forum/23", external_url: "http://google.com")
|
||||||
|
Fabricate(:permalink, url: "/discourse/98", external_url: "http://wikipedia.org")
|
||||||
|
Fabricate(:permalink, url: "/discuss/topic/45", external_url: "http://discourse.org")
|
||||||
|
Fabricate(:permalink, url: "/discuss/topic/76", external_url: "http://try.discourse.org")
|
||||||
|
|
||||||
|
xhr :get, :index, filter: "discourse"
|
||||||
|
|
||||||
|
expect(response).to be_success
|
||||||
|
result = JSON.parse(response.body)
|
||||||
|
expect(result.length).to eq(3)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue