wiki-grav/plugins/git-sync/app/wizard/index.js

380 lines
12 KiB
JavaScript
Raw Permalink Normal View History

import Settings from 'git-sync';
import request from 'admin/utils/request';
import toastr from 'admin/utils/toastr';
import { config } from 'grav-config';
import $ from 'jquery';
import 'whatwg-fetch';
const GIT_REGEX = /(?:git|ssh|https?|git@[-\w.]+):(\/\/)?(.*?)(\.git)(\/?|\#[-\d\w._]+?)$/;
const WIZARD = $('[data-remodal-id="wizard"]');
const RESET_LOCAL = $('[data-remodal-id="reset-local"]');
const SERVICES = { 'github': 'github.com', 'bitbucket': 'bitbucket.org', 'gitlab': 'gitlab.com', 'allothers': 'allothers.repo' };
const BRANCHES = { 'github': 'main', 'bitbucket': 'master', 'gitlab': 'master', 'allothers': 'master' };
const TEMPLATES = {
REPO_URL: 'https://{placeholder}/getgrav/grav.git'
};
const openWizard = () => {
const modal = WIZARD.remodal({ closeOnConfirm: false });
const previous = WIZARD.find('[data-gitsync-action="previous"]');
const next = WIZARD.find('[data-gitsync-action="next"]');
const save = WIZARD.find('[data-gitsync-action="save"]');
STEP = 0;
WIZARD.find(`form > [class^=step-]:not(.step-${STEP}) > .panel`).hide().removeClass('hidden');
WIZARD.find(`form > [class="step-${STEP}"] > .panel`).show();
next.removeClass('hidden');
previous.addClass('hidden');
save.addClass('hidden');
const webhook = $('[name="data[webhook]"]').val();
const webhook_secret = $('[name="data[webhook_secret]"]').val();
$('[name="gitsync[repository]"]').trigger('change');
$('[name="gitsync[webhook]"]').val(webhook);
$('[name="gitsync[webhook_secret]"]').val(webhook_secret);
$('.gitsync-webhook').text(webhook);
modal.open();
};
const disableButton = (next) => {
next
.attr('disabled', 'disabled')
.addClass('hint--top');
};
const enableButton = (next) => {
next
.attr('disabled', null)
.removeClass('hint--top');
};
let STEP = 0;
let STEPS = 0;
let SERVICE = null;
$(document).on('closed', WIZARD, function(e) {
STEP = 0;
});
$(document).on('click', '[data-gitsync-useraction]', (event) => {
event.preventDefault();
const target = $(event.target).closest('[data-gitsync-useraction]');
const action = target.data('gitsyncUseraction');
const URI = `${config.current_url}.json`;
switch (action) {
case 'wizard':
openWizard();
break;
case 'sync':
const relativeURI = target.data('gitsync-uri');
target.find('i').removeClass('fa-cloud fa-git').addClass('fa-circle-o-notch fa-spin');
request(relativeURI || URI, {
method: 'post',
body: { task: 'synchronize' }
}, () => {
target.find('i').removeClass('fa-circle-o-notch fa-spin').addClass(relativeURI ? 'fa-git' : 'fa-cloud');
});
break;
case 'reset':
const modal = RESET_LOCAL.remodal({ closeOnConfirm: false });
modal.open();
if (!RESET_LOCAL.data('_reset_event_set_')) {
RESET_LOCAL.find('[data-gitsync-action="reset-local"]').one('click', () => {
modal.close();
RESET_LOCAL.data('_reset_event_set_', true);
target.find('i').removeClass('fa-history').addClass('fa-circle-o-notch fa-spin');
request(URI, {
method: 'post',
body: { task: 'resetlocal' }
}, () => {
RESET_LOCAL.data('_reset_event_set_', false);
target.find('i').removeClass('fa-circle-o-notch fa-spin').addClass('fa-history');
});
});
}
break;
}
});
$(document).on('click', '[data-gitsync-action]', (event) => {
event.preventDefault();
const target = $(event.target).closest('[data-gitsync-action]');
const previous = WIZARD.find('[data-gitsync-action="previous"]');
const next = WIZARD.find('[data-gitsync-action="next"]');
const save = WIZARD.find('[data-gitsync-action="save"]');
const action = target.data('gitsyncAction');
const user = $('[name="gitsync[repo_user]"]').val();
const noUser = $('[name="gitsync[no_user]"]').is(':checked');
const password = $('[name="gitsync[repo_password]"]').val();
const repository = $('[name="gitsync[repo_url]"]').val();
const branch = $('[name="gitsync[branch]"]').val();
const webhook = $('[name="gitsync[webhook]"]').val();
const webhook_enabled = $('[name="gitsync[webhook_enabled]"]').is(':checked');
const webhook_secret = $('[name="gitsync[webhook_secret]"]').val();
if (target.attr('disabled')) {
return;
}
let error = [];
if (!user && !noUser) {
error.push('Username is missing.');
}
/*
if (!password) {
error.push('Password is missing.');
}
*/
if (!repository) {
error.push('Repository is missing.');
}
if (['save', 'test'].includes(action)) {
target.find('.fa').removeClass(action === 'test' ? 'fa-plug' : 'fa-check').addClass('fa-spin fa-circle-o-notch');
if (error.length) {
toastr.error(error.join('<br />'));
target.find('.fa').removeClass('fa-spin fa-circle-o-notch').addClass(action === 'test' ? 'fa-plug' : 'fa-check');
return false;
}
}
if (action === 'save') {
const folders = $('[name="gitsync[folders]"]:checked').map((i, item) => item.value);
$('[name="data[repository]"]').val(repository);
$('[name="data[no_user]"]').val(noUser ? '1' : '0');
$('[name="data[user]"]').val(user);
$('[name="data[password]"]').val(password);
$('[name="data[branch]"]').val(branch);
$('[name="data[remote][branch]"]').val(branch);
$('[name="data[webhook]"]').val(webhook);
$(`[name="data[webhook_enabled]"][value="${webhook_enabled ? 1 : 0}"]`).prop('checked', true);
$('[name="data[webhook_secret]"]').val(webhook_secret);
const dataFolders = $('[name="data[folders][]"]');
if (dataFolders && dataFolders[0] && dataFolders[0].selectize) {
dataFolders[0].selectize.setValue(folders.toArray());
}
$('[name="task"][value="save"]').trigger('click');
return false;
}
if (action === 'test') {
const URI = `${config.current_url}.json`;
const test = global.btoa(JSON.stringify({
user: noUser ? '' : user,
password,
repository,
branch
}));
request(URI, {
method: 'post',
body: { test, task: 'testConnection' }
});
target.find('.fa').removeClass('fa-spin fa-circle-o-notch').addClass('fa-plug');
return false;
}
WIZARD.find(`.step-${STEP} > .panel`).slideUp();
STEP += action === 'next' ? +1 : -1;
WIZARD.find(`.step-${STEP} > .panel`).slideDown();
save.addClass('hidden');
if (action === 'next') {
previous.removeClass('hidden');
}
if (STEP <= 0) {
previous.addClass('hidden');
enableButton(next);
}
if (STEP > 0) {
next.removeClass('hidden');
}
if (STEP === 1) {
const selectedRepo = $('[name="gitsync[repository]"]:checked');
if (!selectedRepo.length) {
disableButton(next);
} else {
enableButton(next);
}
}
if (STEP === 2) {
const repoURL = $('[name="gitsync[repo_url]"]').val();
if (!repoURL.length || !branch) {
disableButton(next);
} else {
enableButton(next);
}
}
if (STEP === STEPS) {
next.addClass('hidden');
previous.removeClass('hidden');
save.removeClass('hidden');
}
});
$(document).on('input', '[name="gitsync[no_user]"]', (event) => {
const target = $(event.currentTarget);
const user = $('[name="gitsync[repo_user]"]');
if (target.is(':checked')) {
user
.val('')
.prop('disabled', 'disabled')
.attr('placeholder', '<username not required>');
} else {
user
.prop('disabled', null)
.attr('placeholder', 'Username, not email');
}
});
$(document).on('change', '[name="gitsync[repository]"]', () => {
enableButton(WIZARD.find('[data-gitsync-action="next"]'));
});
$(document).on('input', '[name="gitsync[repo_url]"]', (event) => {
const target = $(event.currentTarget);
const value = target.val();
const isGitURL = GIT_REGEX.test(value);
const next = WIZARD.find('[data-gitsync-action="next"]');
target.removeClass('invalid');
if (!isGitURL) {
target.addClass('invalid');
}
if (isGitURL && value.length) {
enableButton(next);
} else {
disableButton(next);
}
});
$(document).on('keyup', '[data-gitsync-uribase] [name="gitsync[webhook]"]', (event) => {
const target = $(event.currentTarget);
const value = target.val();
$('.gitsync-webhook').text(value);
});
$(document).on('keyup', '[data-gitsync-uribase] [name="gitsync[webhook_secret]"]', (event) => {
$('[data-gitsync-uribase] [name="gitsync[webhook_enabled]"]').trigger('change');
});
$(document).on('change', '[data-gitsync-uribase] [name="gitsync[webhook_enabled]"]', (event) => {
const target = $(event.currentTarget);
const checked = target.is(':checked');
const secret = $('[name="gitsync[webhook_secret]"]').val();
target.closest('.webhook-secret-wrapper').find('label:last-child')[checked ? 'removeClass' : 'addClass']('hidden');
$('.gitsync-webhook-secret').html(!checked || !secret.length ? '<em>leave empty</em>' : `<code>${secret}</code>`);
});
$(document).on('change', '[name="gitsync[repository]"]', (event) => {
const target = $(event.target);
if (!target.is(':checked')) {
return;
}
SERVICE = target.val();
Object.keys(SERVICES).forEach((service) => {
WIZARD.find(`.hidden-step-${service}`)[service === SERVICE ? 'removeClass' : 'addClass']('hidden');
if (service === SERVICE) {
WIZARD.find('.webhook-secret-wrapper')[service === 'bitbucket' ? 'addClass' : 'removeClass']('hidden');
WIZARD
.find('input[name="gitsync[repo_url]"][placeholder]')
.attr('placeholder', TEMPLATES.REPO_URL.replace(/\{placeholder\}/, SERVICES[service]))
.end()
.find('input[name="gitsync[branch]"]')
.attr('placeholder', BRANCHES[service])
.val(BRANCHES[service]);
}
});
});
$(document).on('click', '[data-access-tokens-details]', (event) => {
event.preventDefault();
const button = $(event.currentTarget);
const panel = button.closest('.access-tokens').find('.access-tokens-details');
panel.slideToggle(250, () => {
const isVisible = panel.is(':visible');
const icon = button.find('.fa');
icon.removeClass('fa-chevron-down fa-chevron-up').addClass(`fa-chevron-${isVisible ? 'up' : 'down'}`);
});
});
const showNotices = (element) => {
const target = $(element);
const selection = target.val().replace(/\//g, '-');
const column = target.closest('.columns').find('.column:last');
column.find('[class*="description-"]').addClass('hidden');
column.find(`.description-${selection}`).removeClass('hidden').hide().fadeIn({
duration: 250
});
};
$(document).on('input', '[data-remodal-id="wizard"] .step-4 input[type="checkbox"]', (event) => {
const target = $(event.currentTarget);
if (!target.is(':checked')) {
return;
}
showNotices(target);
});
$(document).on('mouseenter', '[data-remodal-id="wizard"] .step-4 .info-desc', (event) => {
const target = $(event.currentTarget).siblings('input[type="checkbox"]');
showNotices(target);
});
$(document).on('mouseleave', '[data-remodal-id="wizard"] .step-4 label', (event) => {
const target = $(event.currentTarget);
const container = target.closest('.columns');
const column = container.find('.column:last-child');
column.find('[class*="description-"]').addClass('hidden');
});
$(document).on('mouseleave', '[data-remodal-id="wizard"] .columns .column:first-child', (event) => {
const target = $(event.currentTarget);
const column = target.siblings('.column');
column.find('[class*="description-"]').addClass('hidden');
});
$(document).ready(() => {
STEPS = WIZARD.find('[class^="step-"]').length - 1;
WIZARD.wrapInner('<form></form>');
RESET_LOCAL.wrapInner('<form></form>');
if (WIZARD.length && (Settings.first_time || !Settings.git_installed)) {
openWizard();
}
});
export default Settings;