Lots and lots of cleanup!
This commit is contained in:
parent
6f326320cc
commit
caf2bde6c4
@ -11,7 +11,7 @@ using YTManager;
|
|||||||
namespace YTManager.Migrations
|
namespace YTManager.Migrations
|
||||||
{
|
{
|
||||||
[DbContext(typeof(MediaDB))]
|
[DbContext(typeof(MediaDB))]
|
||||||
[Migration("20170901083156_initial")]
|
[Migration("20170902220142_initial")]
|
||||||
partial class initial
|
partial class initial
|
||||||
{
|
{
|
||||||
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||||
@ -63,6 +63,9 @@ namespace YTManager.Migrations
|
|||||||
|
|
||||||
b.Property<DateTime>("Uploaded");
|
b.Property<DateTime>("Uploaded");
|
||||||
|
|
||||||
|
b.Property<string>("YTChannelID")
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
b.Property<string>("YTVideoID")
|
b.Property<string>("YTVideoID")
|
||||||
.IsRequired();
|
.IsRequired();
|
||||||
|
|
@ -37,6 +37,7 @@ namespace YTManager.Migrations
|
|||||||
ThumbnailURL = table.Column<string>(type: "text", nullable: false),
|
ThumbnailURL = table.Column<string>(type: "text", nullable: false),
|
||||||
Title = table.Column<string>(type: "text", nullable: false),
|
Title = table.Column<string>(type: "text", nullable: false),
|
||||||
Uploaded = table.Column<DateTime>(type: "timestamp", nullable: false),
|
Uploaded = table.Column<DateTime>(type: "timestamp", nullable: false),
|
||||||
|
YTChannelID = table.Column<string>(type: "text", nullable: false),
|
||||||
YTVideoID = table.Column<string>(type: "text", nullable: false)
|
YTVideoID = table.Column<string>(type: "text", nullable: false)
|
||||||
},
|
},
|
||||||
constraints: table =>
|
constraints: table =>
|
@ -62,6 +62,9 @@ namespace YTManager.Migrations
|
|||||||
|
|
||||||
b.Property<DateTime>("Uploaded");
|
b.Property<DateTime>("Uploaded");
|
||||||
|
|
||||||
|
b.Property<string>("YTChannelID")
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
b.Property<string>("YTVideoID")
|
b.Property<string>("YTVideoID")
|
||||||
.IsRequired();
|
.IsRequired();
|
||||||
|
|
||||||
|
@ -22,6 +22,10 @@ namespace YTManager.Models {
|
|||||||
[Required]
|
[Required]
|
||||||
public string YTVideoID { get; set; }
|
public string YTVideoID { get; set; }
|
||||||
|
|
||||||
|
// Channel this video belongs to.
|
||||||
|
[Required]
|
||||||
|
public string YTChannelID { get; set; }
|
||||||
|
|
||||||
// Thumbnail link
|
// Thumbnail link
|
||||||
[Required]
|
[Required]
|
||||||
public string ThumbnailURL { get; set; }
|
public string ThumbnailURL { get; set; }
|
||||||
|
@ -11,8 +11,6 @@ namespace YTManager.Tasks
|
|||||||
{
|
{
|
||||||
public static void run()
|
public static void run()
|
||||||
{
|
{
|
||||||
Console.WriteLine("Startnig job ...");
|
|
||||||
|
|
||||||
// YT API access key
|
// YT API access key
|
||||||
var youtubeService = new YouTubeService(new Google.Apis.Services.BaseClientService.Initializer()
|
var youtubeService = new YouTubeService(new Google.Apis.Services.BaseClientService.Initializer()
|
||||||
{
|
{
|
||||||
@ -29,20 +27,23 @@ namespace YTManager.Tasks
|
|||||||
// Get all the most recent videos for each channel.
|
// Get all the most recent videos for each channel.
|
||||||
channels.ForEach(ch => {
|
channels.ForEach(ch => {
|
||||||
// Get channel videos from youtube.
|
// Get channel videos from youtube.
|
||||||
var query = youtubeService.Activities.List("snippet");
|
var query = youtubeService.Search.List("snippet");
|
||||||
query.ChannelId = ch.YTChannelID;
|
query.ChannelId = ch.YTChannelID;
|
||||||
|
query.MaxResults = 50;
|
||||||
var response = query.Execute();
|
var response = query.Execute();
|
||||||
|
|
||||||
// Get all videos which aren't already in the DB.
|
// Get all videos which aren't already in the DB.
|
||||||
var notindb = response.Items
|
var notindb = response.Items
|
||||||
.Where(i => !dbcontext.Videos.Any(dbvid => dbvid.YTVideoID != i.Id))
|
.Where(i => i.Id.Kind == "youtube#video")
|
||||||
|
.Where(i => !dbcontext.Videos.Any(dbvid => dbvid.YTVideoID != i.Id.VideoId))
|
||||||
.Select(newvid =>
|
.Select(newvid =>
|
||||||
new Models.Video {
|
new Models.Video {
|
||||||
Title = newvid.Snippet.Title,
|
Title = newvid.Snippet.Title,
|
||||||
Description = newvid.Snippet.Description,
|
Description = newvid.Snippet.Description,
|
||||||
YTVideoID = newvid.Id,
|
YTVideoID = newvid.Id.VideoId,
|
||||||
Uploaded = newvid.Snippet.PublishedAt.GetValueOrDefault(),
|
Uploaded = newvid.Snippet.PublishedAt.GetValueOrDefault(),
|
||||||
AddedtoDB = DateTime.Now,
|
AddedtoDB = DateTime.Now,
|
||||||
|
YTChannelID = newvid.Snippet.ChannelId,
|
||||||
Channel = ch,
|
Channel = ch,
|
||||||
ThumbnailURL = newvid.Snippet.Thumbnails.Medium.Url
|
ThumbnailURL = newvid.Snippet.Thumbnails.Medium.Url
|
||||||
}).ToList();
|
}).ToList();
|
||||||
|
@ -9,86 +9,19 @@
|
|||||||
<link rel="stylesheet" type="text/css" href="index.css">
|
<link rel="stylesheet" type="text/css" href="index.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>Dumb YT Manager</h1>
|
<div class="pageheader">
|
||||||
<p>Youtube banned my account and refuses to say why, taking all my subscribed channels with it. This is a simple scrubscribed channel manager, showing recent releases from each channel and finally proper sub-catagory functionality!</p>
|
<h1>Dumb YT Manager</h1>
|
||||||
|
<p>Youtube banned my account and refuses to say why, taking all my subscribed channels with it. This is a simple scrubscribed channel manager, showing recent releases from each channel and finally proper sub-catagory functionality!</p>
|
||||||
<hr />
|
|
||||||
<h2>Subscribed Channels</h2>
|
|
||||||
<table id="tbody-0">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th>YT Channel ID</th>
|
|
||||||
<th>Title</th>
|
|
||||||
<th>Description</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr v-cloak v-for="entry in entries">
|
|
||||||
<td class="tinytext12px">{{entry.ytChannelID}}</td>
|
|
||||||
<td class="tinytext12px">{{entry.title}}</td>
|
|
||||||
<td class="tinytext12px">{{entry.description}}</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
|
|
||||||
<div id="addchanel-0">
|
|
||||||
<div class="grid-x">
|
|
||||||
<div class="medium-11 cell"></div>
|
|
||||||
<div class="medium-1 cell">
|
|
||||||
<button type="button" class="button input-group" v-on:click='expanded = !expanded'>
|
|
||||||
<i class="fa fa-plus" aria-hidden="true"></i> Add
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<transition name="fade">
|
|
||||||
<form v-cloak v-if="expanded">
|
|
||||||
<h2>Add new Channel</h2>
|
|
||||||
<div class="grid-x">
|
|
||||||
<div class="input-group medium-6 cell">
|
|
||||||
<span class="input-group-label">Title</span>
|
|
||||||
<input class="input-group-field" type="text" placeholder="Title of Entry" v-model="entry.title">
|
|
||||||
</div>
|
|
||||||
<div class="input-group medium-1 cell"></div>
|
|
||||||
<div class="input-group medium-5 cell">
|
|
||||||
<span class="input-group-label">Channel</span>
|
|
||||||
<input class="input-group-field" v-bind:class="{ error: fail }" type="text"
|
|
||||||
@blur="newurl" placeholder="URL of Entry" v-model="entry.yTChannelID">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="input-group">
|
|
||||||
<span class="input-group-label">Description</span>
|
|
||||||
<input class="input-group-field" type="text" placeholder="Description of Entry" v-model="entry.description">
|
|
||||||
</div>
|
|
||||||
<div class="grid-x">
|
|
||||||
<img class="small-4 cell" :src="entry.thumbnailURL" />
|
|
||||||
<div class="small-7"></div>
|
|
||||||
<button type="button" style="max-height:50px" class="button input-group small-1 cell" @click="submit">Submit</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</transition>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<hr />
|
||||||
|
<h2>Subscribed Channels</h2>
|
||||||
|
<div id="subbedchannelstable-0"></div>
|
||||||
|
<div id="addchanel-0"></div>
|
||||||
|
|
||||||
<hr />
|
<hr />
|
||||||
<h2>Videos in DB</h2>
|
<h2>Videos in DB</h2>
|
||||||
<table id="videostb-0">
|
<div id="videosindbtable-0"></div>
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th>Title</th>
|
|
||||||
<th>Description</th>
|
|
||||||
<th>Youtube Video ID</th>
|
|
||||||
<th>ID</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr v-cloak v-for="video in Videos">
|
|
||||||
<td class="tinytext12px">{{video.title}}</td>
|
|
||||||
<td class="tinytext12px">{{video.description}}</td>
|
|
||||||
<td class="tinytext12px">{{video.ytVideoID}}</td>
|
|
||||||
<td class="tinytext12px">{{video.videoId}}</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
|
|
||||||
<!-- Compressed JavaScript -->
|
<!-- Compressed JavaScript -->
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
|
||||||
|
@ -16,6 +16,45 @@ var addchanel0 = new Vue({
|
|||||||
fail: false,
|
fail: false,
|
||||||
expanded: false
|
expanded: false
|
||||||
},
|
},
|
||||||
|
template: `
|
||||||
|
<div>
|
||||||
|
<div class="grid-x">
|
||||||
|
<div class="medium-11 cell"></div>
|
||||||
|
<div class="medium-1 cell">
|
||||||
|
<button type="button" class="button input-group" v-on:click='expanded = !expanded'>
|
||||||
|
<i class="fa fa-plus" aria-hidden="true"></i> Add
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<transition name="fade">
|
||||||
|
<form v-cloak v-if="expanded">
|
||||||
|
<h2>Add new Channel</h2>
|
||||||
|
<div class="grid-x">
|
||||||
|
<div class="input-group medium-6 cell">
|
||||||
|
<span class="input-group-label">Title</span>
|
||||||
|
<input class="input-group-field" type="text" placeholder="Title of Entry" v-model="entry.title">
|
||||||
|
</div>
|
||||||
|
<div class="input-group medium-1 cell"></div>
|
||||||
|
<div class="input-group medium-5 cell">
|
||||||
|
<span class="input-group-label">Channel</span>
|
||||||
|
<input class="input-group-field" v-bind:class="{ error: fail }" type="text"
|
||||||
|
@blur="newurl" placeholder="URL of Entry" v-model="entry.yTChannelID">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="input-group">
|
||||||
|
<span class="input-group-label">Description</span>
|
||||||
|
<input class="input-group-field" type="text" placeholder="Description of Entry" v-model="entry.description">
|
||||||
|
</div>
|
||||||
|
<div class="grid-x">
|
||||||
|
<img class="small-4 cell" :src="entry.thumbnailURL" />
|
||||||
|
<div class="small-7"></div>
|
||||||
|
<button type="button" style="max-height:50px" class="button input-group small-1 cell" @click="submit">Submit</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</transition>
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
methods: {
|
methods: {
|
||||||
submit: function (event) {
|
submit: function (event) {
|
||||||
// Send the channel to our API and request tables be updated.
|
// Send the channel to our API and request tables be updated.
|
||||||
@ -78,10 +117,30 @@ var addchanel0 = new Vue({
|
|||||||
|
|
||||||
// Bind to our table of entries.
|
// Bind to our table of entries.
|
||||||
var tbody0 = new Vue({
|
var tbody0 = new Vue({
|
||||||
el: '#tbody-0',
|
el: '#subbedchannelstable-0',
|
||||||
data: {
|
data: {
|
||||||
entries: [""]
|
entries: [""]
|
||||||
},
|
},
|
||||||
|
template: `
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>YT Channel ID</th>
|
||||||
|
<th>Title</th>
|
||||||
|
<th>Description</th>
|
||||||
|
<th>ID</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr v-cloak v-for="entry in entries">
|
||||||
|
<td class="tinytext12px">{{entry.ytChannelID}}</td>
|
||||||
|
<td class="tinytext12px">{{entry.title}}</td>
|
||||||
|
<td class="tinytext12px">{{entry.description}}</td>
|
||||||
|
<td class="tinytext12px">{{entry.channelId}}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
`,
|
||||||
methods: {
|
methods: {
|
||||||
retrieve: function (event) {
|
retrieve: function (event) {
|
||||||
axios.get('/api/Channels')
|
axios.get('/api/Channels')
|
||||||
@ -106,10 +165,32 @@ var tbody0 = new Vue({
|
|||||||
|
|
||||||
// Grid if images.
|
// Grid if images.
|
||||||
var videostb = new Vue({
|
var videostb = new Vue({
|
||||||
el: '#videostb-0',
|
el: '#videosindbtable-0',
|
||||||
data: {
|
data: {
|
||||||
Videos: []
|
Videos: []
|
||||||
},
|
},
|
||||||
|
template: `
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Title</th>
|
||||||
|
<th>Description</th>
|
||||||
|
<th>Youtube Video ID</th>
|
||||||
|
<th>Uploaded</th>
|
||||||
|
<th>ID</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr v-cloak v-for="video in Videos">
|
||||||
|
<td class="tinytext12px">{{video.title}}</td>
|
||||||
|
<td class="tinytext12px">{{video.description}}</td>
|
||||||
|
<td class="tinytext12px">{{video.ytVideoID}}</td>
|
||||||
|
<td class="tinytext12px">{{video.uploaded}}</td>
|
||||||
|
<td class="tinytext12px">{{video.videoId}}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
`,
|
||||||
methods: {
|
methods: {
|
||||||
// Get new videos from the web api.
|
// Get new videos from the web api.
|
||||||
retrieve: function (event) {
|
retrieve: function (event) {
|
||||||
|
@ -1,12 +1,17 @@
|
|||||||
body {
|
body {
|
||||||
max-width: 1280px;
|
margin-left: 5%;
|
||||||
margin: auto;
|
margin-right: 5%;
|
||||||
line-height: 1.6;
|
line-height: 1.6;
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
color: #444;
|
color: #444;
|
||||||
background-color: #F8F8F8;
|
background-color: #F8F8F8;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.pageheader {
|
||||||
|
max-width: 1280px;
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
|
||||||
.error {
|
.error {
|
||||||
background-color: red;
|
background-color: red;
|
||||||
}
|
}
|
||||||
@ -16,10 +21,6 @@
|
|||||||
margin: auto;
|
margin: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
#vidholder-0 {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
.tinytext10px{ font-size: 10px; }
|
.tinytext10px{ font-size: 10px; }
|
||||||
.tinytext12px{ font-size: 12px; }
|
.tinytext12px{ font-size: 12px; }
|
||||||
.tinytext14px{ font-size: 14px; }
|
.tinytext14px{ font-size: 14px; }
|
||||||
@ -34,6 +35,6 @@
|
|||||||
transition: opacity 0.75s;
|
transition: opacity 0.75s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fade-enter, .fade-leave-to /* .fade-leave-active below version 2.1.8 */ {
|
.fade-enter, .fade-leave-to {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
}
|
}
|
||||||
|
@ -9,24 +9,14 @@
|
|||||||
<link rel="stylesheet" type="text/css" href="index.css">
|
<link rel="stylesheet" type="text/css" href="index.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>Dumb YT Manager</h1>
|
<div class="pageheader">
|
||||||
<p>Youtube banned my account and refuses to say why, taking all my subscribed channels with it. This is a simple scrubscribed channel manager, showing recent releases from each channel and finally proper sub-catagory functionality!</p>
|
<h1>Dumb YT Manager</h1>
|
||||||
|
<p>Youtube banned my account and refuses to say why, taking all my subscribed channels with it. This is a simple scrubscribed channel manager, showing recent releases from each channel and finally proper sub-catagory functionality!</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
<hr />
|
<hr />
|
||||||
<h2>Most Recent Videos</h2>
|
<h2>Most Recent Videos</h2>
|
||||||
<div v-cloak id="vidholder-0">
|
<div v-cloak id="vidholder-0"></div>
|
||||||
<div class="grid-x grid-margin-x imgrow" v-for="row in Videos.Rows">
|
|
||||||
<div class="card medium-2 large-1 cell" v-for="video in row.Columns">
|
|
||||||
<div class="card-divider">
|
|
||||||
<h6>{{video.title}}</h6>
|
|
||||||
</div>
|
|
||||||
<img :src="video.thumbnailURL">
|
|
||||||
<div class="card-section">
|
|
||||||
<p class="tinytext14px">{{video.description}}</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Compressed JavaScript -->
|
<!-- Compressed JavaScript -->
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
|
||||||
|
@ -7,36 +7,32 @@ var maxcharsdescriptionfield = 100;
|
|||||||
var vidholder = new Vue({
|
var vidholder = new Vue({
|
||||||
el: '#vidholder-0',
|
el: '#vidholder-0',
|
||||||
data: {
|
data: {
|
||||||
Videos: {
|
Videos: []
|
||||||
Rows: [
|
|
||||||
/*{
|
|
||||||
Columns: [
|
|
||||||
{
|
|
||||||
"title": "0",
|
|
||||||
"description": "",
|
|
||||||
"yTVideoID": "",
|
|
||||||
"thumbnailURL": "",
|
|
||||||
"uploaded": "",
|
|
||||||
"yTChannelID": ""
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}*/
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
// Template has wrapping div because v-for can't be in root it seems.
|
||||||
|
template: `
|
||||||
|
<div>
|
||||||
|
<div class="grid-x grid-margin-x large-up-6">
|
||||||
|
<div v-for="video in Videos" class="cell"><div class="card">
|
||||||
|
<div class="card-divider">
|
||||||
|
<h6>{{ video.title }}</h6>
|
||||||
|
</div>
|
||||||
|
<a :href="video.url"><img :src="video.thumbnailURL"></a>
|
||||||
|
<div class="card-section">
|
||||||
|
<p class="tinytext14px">{{ video.description }}</p>
|
||||||
|
</div>
|
||||||
|
</div></div>
|
||||||
|
</div >
|
||||||
|
</div>
|
||||||
|
`,
|
||||||
methods: {
|
methods: {
|
||||||
// Get new videos from the web api.
|
// Get new videos from the web api.
|
||||||
retrieve: function (event) {
|
retrieve: function (event) {
|
||||||
// Wipe out all old entries.
|
// Wipe out all old entries.
|
||||||
this.Videos.Rows = [];
|
this.Videos = [];
|
||||||
|
|
||||||
axios.get('/api/Videos')
|
axios.get('/api/Videos')
|
||||||
.then(function (response) {
|
.then(function (response) {
|
||||||
// Dimension counters and limits.
|
|
||||||
var maxcols = 4;
|
|
||||||
var colcounter = 0;
|
|
||||||
var rowcounter = 0;
|
|
||||||
|
|
||||||
// And fill it with all the retrieved entries.
|
// And fill it with all the retrieved entries.
|
||||||
response.data.forEach(function (x) {
|
response.data.forEach(function (x) {
|
||||||
// Trim description if needed.
|
// Trim description if needed.
|
||||||
@ -44,21 +40,11 @@ var vidholder = new Vue({
|
|||||||
x.description = x.description.substring(0, maxcharsdescriptionfield) + " ...";
|
x.description = x.description.substring(0, maxcharsdescriptionfield) + " ...";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle creation of a new row.
|
// Generate a new URL by adding the YT ID.
|
||||||
if (colcounter == 0) {
|
x.url = "https://www.youtube.com/watch?v=" + x.ytVideoID
|
||||||
Vue.set(this.Videos.Rows, rowcounter, []);
|
|
||||||
Vue.set(this.Videos.Rows[rowcounter], 'Columns', []);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add it to our array
|
// Add it to our array
|
||||||
this.Videos.Rows[rowcounter].Columns.push(x);
|
this.Videos.push(x);
|
||||||
|
|
||||||
// So we get it all in groups of maxcols.
|
|
||||||
colcounter = colcounter + 1;
|
|
||||||
if (colcounter == maxcols) {
|
|
||||||
colcounter = 0;
|
|
||||||
rowcounter = rowcounter + 1;
|
|
||||||
}
|
|
||||||
}.bind(this));
|
}.bind(this));
|
||||||
}.bind(this))
|
}.bind(this))
|
||||||
.catch(function (error) {
|
.catch(function (error) {
|
||||||
|
Loading…
Reference in New Issue
Block a user