You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Hydro/ui/pages/problem_detail.page.js

298 lines
7.7 KiB
JavaScript

import Tether from 'tether';
import { NamedPage } from 'vj/misc/PageLoader';
import Navigation from 'vj/components/navigation';
import Notification from 'vj/components/notification';
import { ActionDialog } from 'vj/components/dialog';
import DomainSelectAutoComplete from 'vj/components/autocomplete/DomainSelectAutoComplete';
import loadReactRedux from 'vj/utils/loadReactRedux';
import delay from 'vj/utils/delay';
import request from 'vj/utils/request';
import i18n from 'vj/utils/i18n';
class ProblemPageExtender {
constructor() {
this.isExtended = false;
this.inProgress = false;
this.$content = $('.problem-content-container');
this.$contentBound = this.$content.closest('.section');
this.$scratchpadContainer = $('.scratchpad-container');
}
async extend() {
if (this.inProgress) {
return;
}
if (this.isExtended) {
return;
}
this.inProgress = true;
const bound = this.$contentBound
.get(0)
.getBoundingClientRect();
this.$content.transition({ opacity: 0 }, { duration: 100 });
await delay(100);
Navigation.instance.floating.set('scratchpad', true);
Navigation.instance.logoVisible.set('scratchpad', true);
Navigation.instance.expanded.set('scratchpad', true);
$('body').addClass('header--collapsed mode--scratchpad');
await this.$scratchpadContainer
.css({
left: bound.left,
top: bound.top,
width: bound.width,
height: bound.height,
})
.show()
.transition({
left: 0,
top: 0,
width: '100%',
height: '100%',
}, {
duration: 500,
easing: 'easeOutCubic',
})
.promise();
$('.main > .row').hide();
$('.footer').hide();
$(window).scrollTop(0);
this.inProgress = false;
this.isExtended = true;
}
async collapse() {
if (this.inProgress) {
return;
}
if (!this.isExtended) {
return;
}
this.inProgress = true;
$(window).scrollTop(0);
$('.main > .row').show();
$('.footer').show();
const bound = this.$contentBound
.get(0)
.getBoundingClientRect();
Navigation.instance.floating.set('scratchpad', false);
Navigation.instance.logoVisible.set('scratchpad', false);
Navigation.instance.expanded.set('scratchpad', false);
$('body').removeClass('header--collapsed mode--scratchpad');
await this.$scratchpadContainer
.transition({
left: bound.left,
top: bound.top,
width: bound.width,
height: bound.height,
}, {
duration: 500,
easing: 'easeOutCubic',
})
.promise();
this.$scratchpadContainer.hide();
this.$content.transition({ opacity: 1 }, { duration: 100 });
this.inProgress = false;
this.isExtended = false;
}
toggle() {
if (this.isExtended) {
this.collapse();
} else {
this.extend();
}
}
}
const page = new NamedPage(['problem_detail', 'contest_detail_problem', 'homework_detail_problem'], () => {
let reactLoaded = false;
let $floatingSidebar = null;
let renderReact = null;
let unmountReact = null;
const extender = new ProblemPageExtender();
async function scratchpadFadeIn() {
await $('#scratchpad')
.transition({
opacity: 1,
}, {
duration: 200,
easing: 'easeOutCubic',
})
.promise();
}
async function scratchpadFadeOut() {
await $('#scratchpad')
.transition({
opacity: 0,
}, {
duration: 200,
easing: 'easeOutCubic',
})
.promise();
}
function updateFloatingSidebar() {
$floatingSidebar.tether.position();
}
async function createSidebar() {
$floatingSidebar = $('.section--problem-sidebar')
.clone()
.addClass('scratchpad__sidebar visible')
.appendTo('body');
$floatingSidebar.find('a').attr('target', '_blank');
$floatingSidebar.tether = new Tether({
element: $floatingSidebar,
offset: '-20px 20px',
target: '.scratchpad__problem',
attachment: 'top right',
targetAttachment: 'top right',
});
await delay(100);
$floatingSidebar.tether.position();
}
async function removeSidebar() {
$floatingSidebar.tether.destroy();
$floatingSidebar.remove();
$floatingSidebar = null;
}
async function loadReact() {
if (reactLoaded) {
return;
}
$('.loader-container').show();
const SockJs = await import('sockjs-client');
const { default: ScratchpadApp } = await import('../components/scratchpad');
const { default: ScratchpadReducer } = await import('../components/scratchpad/reducers');
const {
React, render, unmountComponentAtNode, Provider, store,
} = await loadReactRedux(ScratchpadReducer);
const sock = new SockJs(Context.socketUrl);
sock.onmessage = (message) => {
const msg = JSON.parse(message.data);
store.dispatch({
type: 'SCRATCHPAD_RECORDS_PUSH',
payload: msg,
});
};
renderReact = () => {
render(
<Provider store={store}>
<ScratchpadApp />
</Provider>,
$('#scratchpad').get(0),
);
};
unmountReact = () => {
unmountComponentAtNode($('#scratchpad').get(0));
};
reactLoaded = true;
$('.loader-container').hide();
}
const domainSelector = DomainSelectAutoComplete.getOrConstruct($('.dialog__body--copy-to [name="domain_id"]'));
const copyProblemToDialog = new ActionDialog({
$body: $('.dialog__body--copy-to > div'),
onDispatch(action) {
if (action === 'ok' && domainSelector.value() === null) {
domainSelector.focus();
return false;
}
return true;
},
});
copyProblemToDialog.clear = function () {
domainSelector.clear();
return this;
};
async function handleClickCopyProblem() {
const action = await copyProblemToDialog.clear().open();
if (action !== 'ok') {
return;
}
const domainId = copyProblemToDialog.$dom.find('[name="domain_id"]').val();
const useNumericId = copyProblemToDialog.$dom.find('[name="numeric_pid"]').prop('checked');
const isHidden = copyProblemToDialog.$dom.find('[name="hidden"]').prop('checked');
const payload = {
operation: 'copy_to_domain',
dest_domain_id: domainId,
};
if (useNumericId) {
payload.numeric_pid = 'on';
}
if (isHidden) {
payload.hidden = 'on';
}
try {
const data = await request.post('', payload);
Notification.success(i18n('Problem is successfully copied.'));
await delay(1000);
window.location.href = data.new_problem_url;
} catch (error) {
Notification.error(error.message);
}
}
async function enterScratchpadMode() {
await extender.extend();
await loadReact();
renderReact();
await scratchpadFadeIn();
await createSidebar();
}
async function leaveScratchpadMode() {
await removeSidebar();
await scratchpadFadeOut();
$('.problem-content-container').append($('.problem-content'));
await extender.collapse();
unmountReact();
}
$(document).on('vjScratchpadRelayout', updateFloatingSidebar);
$(document).on('click', '[name="problem-sidebar__open-scratchpad"]', (ev) => {
enterScratchpadMode();
ev.preventDefault();
});
$(document).on('click', '[name="problem-sidebar__quit-scratchpad"]', (ev) => {
leaveScratchpadMode();
ev.preventDefault();
});
$(document).on('click', '[name="problem-sidebar__copy-to"]', (ev) => {
handleClickCopyProblem();
ev.preventDefault();
});
$(document).on('click', '[name="problem-sidebar__show-category"]', (ev) => {
$(ev.currentTarget).hide();
$('[name="problem-sidebar__categories"]').show();
});
});
export default page;