meos-2024/code/TabClass.cpp
2017-10-29 18:48:54 +01:00

4066 lines
125 KiB
C++

/************************************************************************
MeOS - Orienteering Software
Copyright (C) 2009-2017 Melin Software HB
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
Melin Software HB - software@melin.nu - www.melin.nu
Eksoppsvägen 16, SE-75646 UPPSALA, Sweden
************************************************************************/
#include "stdafx.h"
#include <cassert>
#include "resource.h"
#include <commctrl.h>
#include <commdlg.h>
#include <algorithm>
#include "oEvent.h"
#include "xmlparser.h"
#include "gdioutput.h"
#include "csvparser.h"
#include "SportIdent.h"
#include "meos_util.h"
#include "oListInfo.h"
#include "TabClass.h"
#include "ClassConfigInfo.h"
#include "meosException.h"
#include "gdifonts.h"
#include "oEventDraw.h"
#include "MeOSFeatures.h"
extern pEvent gEvent;
const char *visualDrawWindow = "visualdraw";
TabClass::TabClass(oEvent *poe):TabBase(poe)
{
handleCloseWindow.tabClass = this;
clearCompetitionData();
}
void TabClass::clearCompetitionData() {
pSettings.clear();
pSavedDepth = 3600;
pFirstRestart = 3600;
pTimeScaling = 1.0;
pInterval = 120;
currentStage = -1;
EditChanged = false;
ClassId=0;
tableMode = false;
showForkingGuide = false;
storedNStage = L"3";
storedStart = L"";
storedPredefined = oEvent::PredefinedTypes(-1);
cInfoCache.clear();
hasWarnedDirect = false;
hasWarnedStartTime = false;
lastSeedMethod = -1;
lastSeedPreventClubNb = true;
lastSeedReverse = false;
lastSeedGroups = L"1";
lastPairSize = 1;
lastFirstStart = L"";
lastInterval = L"2:00";
lastNumVac = L"0";
lastHandleBibs = false;
lastScaleFactor = L"1.0";
lastMaxAfter = L"60:00";
gdioutput *gdi = getExtraWindow(visualDrawWindow, false);
if (gdi) {
gdi->closeWindow();
}
}
TabClass::~TabClass(void)
{
}
bool ClassInfoSortStart(ClassInfo &ci1, ClassInfo &ci2)
{
return ci1.firstStart>ci2.firstStart;
}
void TabClass::HandleCloseWindow::handle(gdioutput &gdi, BaseInfo &info, GuiEventType type) {
if (type == GUI_EVENT) {
EventInfo &ei = dynamic_cast<EventInfo &>(info);
if (ei.id == "CloseWindow") {
tabClass->closeWindow(gdi);
}
}
}
void TabClass::closeWindow(gdioutput &gdi) {
}
int ClassesCB(gdioutput *gdi, int type, void *data)
{
TabClass &tc = dynamic_cast<TabClass &>(*gdi->getTabs().get(TClassTab));
return tc.classCB(*gdi, type, data);
}
int MultiCB(gdioutput *gdi, int type, void *data)
{
TabClass &tc = dynamic_cast<TabClass &>(*gdi->getTabs().get(TClassTab));
return tc.multiCB(*gdi, type, data);
}
int DrawClassesCB(gdioutput *gdi, int type, void *data)
{
TabClass &tc = dynamic_cast<TabClass &>(*gdi->getTabs().get(TClassTab));
if (type == GUI_LISTBOX) {
tc.enableLoadSettings(*gdi);
}
else if (type == GUI_INPUT || type == GUI_INPUTCHANGE) {
InputInfo &ii = *(InputInfo *)data;
if (ii.id.length() > 1) {
int id = atoi(ii.id.substr(1).c_str());
if (id > 0 && ii.changed()) {
string key = "C" + itos(id);
TextInfo &ti = dynamic_cast<TextInfo&>(gdi->getBaseInfo(key.c_str()));
ti.setColor(colorRed);
gdi->enableInput("DrawAdjust");
gdi->refreshFast();
}
}
}
return 0;
}
int TabClass::multiCB(gdioutput &gdi, int type, void *data)
{
if (type==GUI_BUTTON) {
ButtonInfo bi=*(ButtonInfo *)data;
if (bi.id=="ChangeLeg") {
gdi.dropLine();
legSetup(gdi);
gdi.refresh();
}
else if (bi.id == "CommonStart") {
gdi.setInputStatus("CommonStartTime", gdi.isChecked(bi.id));
}
else if (bi.id == "CoursePool") {
string strId = "StageCourses_label";
gdi.setTextTranslate(strId, getCourseLabel(gdi.isChecked(bi.id)), true);
setLockForkingState(gdi, gdi.isChecked("CoursePool"), gdi.isChecked("LockForking"));
}
else if (bi.id == "LockForking") {
setLockForkingState(gdi, gdi.isChecked("CoursePool"), gdi.isChecked(bi.id));
}
else if (bi.id == "DefineForking") {
if (!checkClassSelected(gdi))
return false;
save(gdi, true);
EditChanged=true;
defineForking(gdi, true);
return true;
}
else if (bi.id == "ApplyForking") {
showForkingGuide = false;
pClass pc = oe->getClass(ClassId);
vector<pRunner> allR;
oe->getRunners(ClassId, 0, allR, false);
bool doClear = false;
for (size_t k = 0; k < allR.size(); k++) {
if (allR[k]->getCourseId() != 0) {
if (!doClear) {
if (gdi.ask(L"Vill du nollställa alla manuellt tilldelade banor?"))
doClear = true;
else
break;
}
if (doClear)
allR[k]->setCourseId(0);
}
}
pair<int,int> res = pc->autoForking(forkingSetup);
gdi.alert("Created X distinct forkings using Y courses.#" +
itos(res.first) + "#" + itos(res.second));
loadPage(gdi);
EditChanged=true;
}
else if (bi.id == "AssignCourses") {
set<int> selectedCourses, selectedLegs;
gdi.getSelection("AllCourses", selectedCourses);
gdi.getSelection("AllStages", selectedLegs);
for (set<int>::iterator it = selectedLegs.begin(); it != selectedLegs.end(); ++it) {
int leg = *it;
forkingSetup[leg].clear();
forkingSetup[leg].insert(forkingSetup[leg].begin(), selectedCourses.begin(), selectedCourses.end());
}
bool empty = true;
for (size_t k = 0; k < forkingSetup.size(); k++) {
if (forkingSetup[k].empty()) {
gdi.setText("leg"+ itos(k), lang.tl("Leg X: Do not modify.#" + itos(k+1)));
}
else {
empty = false;
wstring crs;
for (size_t j = 0; j < forkingSetup[k].size(); j++) {
if (j>0)
crs += L", ";
crs += oe->getCourse(forkingSetup[k][j])->getName();
if (j > 3) {
crs += L"...";
break;
}
}
gdi.setText("leg"+ itos(k), lang.tl(L"Leg X: Use Y.#" + itow(k+1) + L"#" + crs));
}
}
gdi.setInputStatus("ApplyForking", !empty);
gdi.setSelection("AllCourses", set<int>());
gdi.setSelection("AllStages", set<int>());
gdi.refresh();
}
else if (bi.id == "ClearCourses") {
gdi.setSelection("AllCourses", set<int>());
gdi.setSelection("AllStages", set<int>());
gdi.disableInput("AssignCourses");
}
else if (bi.id == "ShowForking") {
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Klassen finns ej.");
vector< vector<int> > forks;
set< pair<int, int> > unfairLegs;
vector< vector<int> > legOrder;
pc->checkForking(legOrder, forks, unfairLegs);
gdioutput *gdi_new = getExtraWindow("fork", true);
wstring title = lang.tl(L"Forkings for X#" + pc->getName());
if (!gdi_new)
gdi_new = createExtraWindow("fork", title,
gdi.scaleLength(1024) );
else
gdi_new->setWindowTitle(title);
gdi_new->clearPage(false);
gdi_new->addString("", fontMediumPlus, "Forkings");
for (size_t k = 0; k < forks.size(); k++) {
gdi_new->dropLine(0.7);
wstring ver = itow(k+1) + L": ";
for (size_t j = 0; j < legOrder[k].size(); j++) {
pCourse crs = oe->getCourse(legOrder[k][j]);
if (crs) {
if (j>0)
ver += L", ";
ver += crs->getName();
}
}
gdi_new->addStringUT(1, ver);
gdi_new->pushX();
gdi_new->fillRight();
for (size_t j = 0; j < forks[k].size(); j++) {
wstring ctrl;
if (forks[k][j] > 0)
ctrl += itow(forks[k][j]);
else {
if (j == 0)
ctrl += lang.tl("Start");
else if (j + 1 == forks[k].size())
ctrl += lang.tl("Mål");
else
ctrl += lang.tl("Växel");
}
int next = -100;
if (j + 1 < forks[k].size()) {
ctrl += L",";
next = forks[k][j+1];
}
int prev = j>0 ? forks[k][j-1] : -100;
bool warn = unfairLegs.count(make_pair(prev, forks[k][j])) != 0;// ||
//unfairLegs.count(make_pair(forks[k][j], next)) != 0;
TextInfo &ti = gdi_new->addStringUT(italicText, ctrl);
if (warn) {
ti.setColor(colorRed);
ti.format = boldText;
}
gdi.setCX(gdi.getCX() - gdi.scaleLength(4));
}
gdi_new->popX();
gdi_new->fillDown();
gdi_new->dropLine();
}
if (!unfairLegs.empty()) {
gdi_new->dropLine();
gdi_new->addString("", fontMediumPlus, "Unfair control legs");
gdi_new->dropLine(0.5);
for (set< pair<int, int> >::const_iterator p = unfairLegs.begin();
p != unfairLegs.end(); ++p) {
wstring f = p->first > 0 ? itow(p->first) : lang.tl("Växel");
wstring s = p->second > 0 ? itow(p->second) : lang.tl("Växel");
gdi_new->addStringUT(0, makeDash(f + L" - " + s));
}
}
gdi_new->dropLine();
gdi_new->addButton("CloseWindow", "Stäng", ClassesCB);
gdi_new->refresh();
}
else if (bi.id == "OneCourse") {
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Klassen finns ej.");
pc->setNumStages(0);
pc->synchronize();
gdi.restore();
gdi.enableInput("MultiCourse", true);
gdi.enableInput("Courses");
}
else if (bi.id=="SetNStage") {
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Klassen finns ej.");
int total, finished, dns;
oe->getNumClassRunners(pc->getId(), 0, total, finished, dns);
oEvent::PredefinedTypes newType = oEvent::PredefinedTypes(gdi.getSelectedItem("Predefined").first);
int nstages = gdi.getTextNo("NStage");
if (finished > 0) {
if (gdi.ask(L"warning:has_results") == false)
return false;
}
else if (total>0) {
bool ok = false;
ClassType ct = pc->getClassType();
if (ct == oClassIndividual) {
switch (newType) {
case oEvent::PPatrolOptional:
case oEvent::PPool:
case oEvent::PPoolDrawn:
case oEvent::PNoMulti:
case oEvent::PForking:
ok = true;
break;
case oEvent::PNoSettings:
ok = (nstages == 1);
}
}
if (!ok) {
if (gdi.ask(L"warning:has_entries") == false)
return false;
}
}
storedPredefined = newType;
if (nstages > 0)
storedNStage = gdi.getText("NStage");
else {
storedNStage = L"";
if (newType != oEvent::PNoMulti)
nstages = 1; //Fixed by type
}
if (nstages>0 && nstages<41) {
wstring st=gdi.getText("StartTime");
int nst = oe->convertAbsoluteTime(st);
if (warnDrawStartTime(gdi, nst)) {
nst = 3600;
st = oe->getAbsTime(nst);
}
if (nst>0)
storedStart = st;
save(gdi, false); //Clears and reloads
gdi.selectItemByData("Courses", -2);
gdi.disableInput("Courses");
oe->setupRelay(*pc, newType, nstages, st);
if (gdi.hasField("MAdd")) {
gdi.enableInput("MAdd");
gdi.enableInput("MCourses");
gdi.enableInput("MRemove");
}
pc->forceShowMultiDialog(true);
selectClass(gdi, pc->getId());
}
else if (nstages==0){
pc->setNumStages(0);
pc->synchronize();
gdi.restore();
gdi.enableInput("MultiCourse", true);
gdi.enableInput("Courses");
oe->adjustTeamMultiRunners(pc);
}
else {
gdi.alert("Antalet sträckor måste vara ett heltal mellan 0 och 40.");
}
}
else if (bi.id.substr(0, 7)=="@Course") {
int cnr=atoi(bi.id.substr(7).c_str());
selectCourses(gdi, cnr);
}
else if (bi.id=="MAdd"){
DWORD cid=ClassId;
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(cid);
if (!pc)
return false;
if (currentStage>=0){
ListBoxInfo lbi;
if (gdi.getSelectedItem("MCourses", lbi)) {
int courseid=lbi.data;
pc->addStageCourse(currentStage, courseid);
pc->fillStageCourses(gdi, currentStage, "StageCourses");
pc->synchronize();
oe->checkOrderIdMultipleCourses(cid);
}
}
EditChanged=true;
}
else if (bi.id=="MRemove"){
if (!checkClassSelected(gdi))
return false;
DWORD cid=ClassId;
pClass pc=oe->getClass(cid);
if (!pc)
return false;
if (currentStage>=0){
ListBoxInfo lbi;
if (gdi.getSelectedItem("StageCourses", lbi)) {
int courseid=lbi.data;
pc->removeStageCourse(currentStage, courseid, lbi.index);
pc->synchronize();
pc->fillStageCourses(gdi, currentStage, "StageCourses");
}
}
}
EditChanged=true;
}
else if (type==GUI_LISTBOX) {
ListBoxInfo bi=*(ListBoxInfo *)data;
if (bi.id.substr(0, 7)=="LegType") {
LegTypes lt = LegTypes(bi.data);
int i=atoi(bi.id.substr(7).c_str());
pClass pc=oe->getClass(ClassId);
if (!pc)
return false;
if (lt == pc->getLegType(i))
return 0;
pc->setLegType(i, lt);
char legno[10];
sprintf_s(legno, "%d", i);
updateStartData(gdi, pc, i, true, false);
gdi.setInputStatus(string("Restart")+legno, !pc->restartIgnored(i), true);
gdi.setInputStatus(string("RestartRope")+legno, !pc->restartIgnored(i), true);
EditChanged=true;
}
else if (bi.id == "AllStages") {
set<int> t;
gdi.getSelection(bi.id, t);
gdi.setInputStatus("AssignCourses", !t.empty());
}
else if (bi.id.substr(0, 9)=="StartType") {
StartTypes st=StartTypes(bi.data);
int i=atoi(bi.id.substr(9).c_str());
pClass pc=oe->getClass(ClassId);
if (!pc)
return false;
pc->setStartType(i, st, true);
char legno[10];
sprintf_s(legno, "%d", i);
updateStartData(gdi, pc, i, true, false);
gdi.setInputStatus(string("Restart")+legno, !pc->restartIgnored(i), true);
gdi.setInputStatus(string("RestartRope")+legno, !pc->restartIgnored(i), true);
EditChanged=true;
}
else if (bi.id == "Predefined") {
bool nleg;
bool start;
oe->setupRelayInfo(oEvent::PredefinedTypes(bi.data), nleg, start);
gdi.setInputStatus("NStage", nleg);
gdi.setInputStatus("StartTime", start);
wstring nl = gdi.getText("NStage");
if (!nleg && _wtoi(nl.c_str()) != 0) {
storedNStage = nl;
gdi.setText("NStage", makeDash(L"-"));
}
else if (nleg && _wtoi(nl.c_str()) == 0) {
gdi.setText("NStage", storedNStage);
}
wstring st = gdi.getText("StartTime");
if (!start && _wtoi(nl.c_str()) != 0) {
storedStart = st;
gdi.setText("StartTime", makeDash(L"-"));
}
else if (start && _wtoi(nl.c_str()) == 0) {
gdi.setText("StartTime", storedStart);
}
}
else if (bi.id=="Courses") {
EditChanged=true;
}
}
else if (type==GUI_INPUTCHANGE){
InputInfo ii=*(InputInfo *)data;
EditChanged=true;
if (ii.id=="NStage")
gdi.enableInput("SetNStage");
//else if (ii.id=="")
}
return 0;
}
int TabClass::classCB(gdioutput &gdi, int type, void *data)
{
if (type==GUI_BUTTON) {
ButtonInfo bi=*(ButtonInfo *)data;
if (bi.id=="Cancel") {
showForkingGuide = false;
loadPage(gdi);
return 0;
}
else if (bi.id == "UseAdvanced") {
bool checked = gdi.isChecked("UseAdvanced");
oe->setProperty("AdvancedClassSettings", checked);
save(gdi, true);
PostMessage(gdi.getHWNDTarget(), WM_USER + 2, TClassTab, 0);
}
else if (bi.id=="SwitchMode") {
if (!tableMode)
save(gdi, true);
tableMode=!tableMode;
loadPage(gdi);
}
else if (bi.id=="Restart") {
save(gdi, true);
clearPage(gdi, true);
gdi.addString("", 2, "Omstart i stafettklasser");
gdi.addString("", 10, "help:31661");
gdi.addListBox("RestartClasses", 200, 250, 0, L"Stafettklasser", L"", true);
oe->fillClasses(gdi, "RestartClasses", oEvent::extraNone, oEvent::filterOnlyMulti);
gdi.pushX();
gdi.fillRight();
oe->updateComputerTime();
int t=oe->getComputerTime()-(oe->getComputerTime()%60)+60;
gdi.addInput("Rope", oe->getAbsTime(t), 6, 0, L"Repdragningstid");
gdi.addInput("Restart", oe->getAbsTime(t+600), 6, 0, L"Omstartstid");
gdi.dropLine(0.9);
gdi.addButton("DoRestart","OK", ClassesCB);
gdi.addButton("Cancel","Stäng", ClassesCB);
gdi.fillDown();
gdi.dropLine(3);
gdi.popX();
}
else if (bi.id=="DoRestart") {
set<int> cls;
gdi.getSelection("RestartClasses", cls);
gdi.fillDown();
set<int>::iterator it;
wstring ropeS=gdi.getText("Rope");
int rope = oe->getRelativeTime(ropeS);
wstring restartS=gdi.getText("Restart");
int restart = oe->getRelativeTime(restartS);
if (rope<=0) {
gdi.alert("Ogiltig repdragningstid.");
return 0;
}
if (restart<=0) {
gdi.alert("Ogiltig omstartstid.");
return 0;
}
if (restart<rope) {
gdi.alert("Repdragningstiden måste ligga före omstartstiden.");
return 0;
}
gdi.addString("", 0, L"Sätter reptid (X) och omstartstid (Y) för:#" +
oe->getAbsTime(rope) + L"#" + oe->getAbsTime(restart));
for (it=cls.begin(); it!=cls.end(); ++it) {
pClass pc=oe->getClass(*it);
if (pc) {
gdi.addStringUT(0, pc->getName());
int ns=pc->getNumStages();
for (int k=0;k<ns;k++) {
pc->setRopeTime(k, ropeS);
pc->setRestartTime(k, restartS);
}
}
}
gdi.scrollToBottom();
gdi.refresh();
}
else if (bi.id=="SaveDrawSettings") {
readClassSettings(gdi);
for(size_t k=0; k<cInfo.size(); k++) {
const ClassInfo &ci = cInfo[k];
if (ci.pc) {
// Save settings with class
ci.pc->synchronize(false);
ci.pc->setDrawFirstStart(drawInfo.firstStart + drawInfo.baseInterval * ci.firstStart);
ci.pc->setDrawInterval(ci.interval * drawInfo.baseInterval);
ci.pc->setDrawVacant(ci.nVacant);
ci.pc->setDrawNumReserved(ci.nExtra);
ci.pc->synchronize(true);
}
}
}
else if (bi.id=="DoDrawAll") {
readClassSettings(gdi);
int method = gdi.getSelectedItem("Method").first;
bool soft = method == DMSOFT;
int pairSize = gdi.getSelectedItem("PairSize").first;
bool drawCoursebased = drawInfo.coursesTogether;
int maxST = 0;
map<int, vector<ClassDrawSpecification> > specs;
for(size_t k=0; k<cInfo.size(); k++) {
const ClassInfo &ci=cInfo[k];
if (ci.pc) {
// Save settings with class
ci.pc->synchronize(false);
int stloc = drawInfo.firstStart + drawInfo.baseInterval * ci.firstStart;
maxST = max(maxST, stloc);
ci.pc->setDrawFirstStart(stloc);
ci.pc->setDrawInterval(ci.interval * drawInfo.baseInterval);
ci.pc->setDrawVacant(ci.nVacant);
ci.pc->setDrawNumReserved(ci.nExtra);
ci.pc->synchronize(true);
}
ClassDrawSpecification cds(ci.classId, 0, drawInfo.firstStart + drawInfo.baseInterval * ci.firstStart,
drawInfo.baseInterval * ci.interval, ci.nVacant);
if (drawCoursebased) {
pCourse pCrs = oe->getClass(ci.classId)->getCourse();
int id = pCrs ? pCrs->getId() : 101010101 + ci.classId;
specs[id].push_back(cds);
}
else
specs[ci.classId].push_back(cds);
}
if (warnDrawStartTime(gdi, maxST))
return 0;
for (map<int, vector<ClassDrawSpecification> >::iterator it = specs.begin();
it != specs.end(); ++it) {
oe->drawList(it->second, soft, pairSize, oEvent::drawAll);
}
oe->addAutoBib();
clearPage(gdi, false);
gdi.addButton("Cancel", "Återgå", ClassesCB);
oListParam par;
oListInfo info;
par.listCode = EStdStartList;
for (size_t k=0; k<cInfo.size(); k++)
par.selection.insert(cInfo[k].classId);
oe->generateListInfo(par, gdi.getLineHeight(), info);
oe->generateList(gdi, false, info, true);
gdi.refresh();
}
else if (bi.id == "RemoveVacant") {
if (gdi.ask(L"Vill du radera alla vakanser från tävlingen?")) {
oe->removeVacanies(0);
gdi.disableInput(bi.id.c_str());
}
}
else if (bi.id == "SelectAll") {
set<int> lst;
oe->getAllClasses(lst);
gdi.setSelection("Classes", lst);
enableLoadSettings(gdi);
}
else if (bi.id == "SelectUndrawn") {
set<int> lst;
oe->getNotDrawnClasses(lst, false);
gdi.setSelection("Classes", lst);
enableLoadSettings(gdi);
}
else if (bi.id == "SelectStart") {
int id = bi.getExtraInt();
vector<int> blocks;
vector<wstring> starts;
oe->getStartBlocks(blocks, starts);
if (size_t(id) < starts.size()) {
wstring start = starts[id];
set<int> lst;
vector<pClass> cls;
oe->getClasses(cls, true);
for (size_t k = 0; k < cls.size(); k++) {
if (cls[k]->getStart() == start)
lst.insert(cls[k]->getId());
}
gdi.setSelection("Classes", lst);
}
enableLoadSettings(gdi);
}
else if (bi.id == "QuickSettings") {
save(gdi, false);
prepareForDrawing(gdi);
}
else if (bi.id == "DrawMode") {
save(gdi, false);
ClassId = 0;
EditChanged=false;
clearPage(gdi, true);
gdi.addString("", boldLarge, "Lotta flera klasser");
gdi.dropLine();
gdi.pushX();
gdi.fillRight();
gdi.addInput("FirstStart", oe->getAbsTime(3600), 10, 0, L"Första (ordinarie) start:");
gdi.addInput("MinInterval", L"2:00", 10, 0, L"Minsta startintervall:");
gdi.fillDown();
gdi.addInput("Vacances", L"5%", 10, 0, L"Andel vakanser:");
gdi.popX();
gdi.addSelection("Method", 200, 200, 0, L"Metod:");
gdi.addItem("Method", lang.tl("Lottning"), DMRandom);
gdi.addItem("Method", lang.tl("SOFT-lottning"), DMSOFT);
gdi.selectItemByData("Method", getDefaultMethod(false));
gdi.fillDown();
gdi.addCheckbox("LateBefore", "Efteranmälda före ordinarie");
gdi.dropLine();
gdi.popX();
gdi.fillRight();
gdi.addButton("AutomaticDraw", "Automatisk lottning", ClassesCB).setDefault();
gdi.addButton("DrawAll", "Manuell lottning", ClassesCB).setExtra(1);
gdi.addButton("Simultaneous", "Gemensam start", ClassesCB);
const bool multiDay = oe->hasPrevStage();
if (multiDay)
gdi.addButton("Pursuit", "Hantera jaktstart", ClassesCB);
gdi.addButton("Cancel", "Återgå", ClassesCB).setCancel();
gdi.dropLine(3);
gdi.newColumn();
gdi.addString("", 10, "help_autodraw");
}
else if (bi.id == "Pursuit") {
pursuitDialog(gdi);
}
else if (bi.id == "SelectAllNoneP") {
bool select = bi.getExtraInt() != 0;
for (int k = 0; k < oe->getNumClasses(); k++) {
gdi.check("PLUse" + itos(k), select);
gdi.setInputStatus("First" + itos(k), select);
}
}
else if (bi.id == "DoPursuit" || bi.id=="CancelPursuit" || bi.id == "SavePursuit") {
bool cancel = bi.id=="CancelPursuit";
int maxAfter = convertAbsoluteTimeMS(gdi.getText("MaxAfter"));
int deltaRestart = convertAbsoluteTimeMS(gdi.getText("TimeRestart"));
int interval = convertAbsoluteTimeMS(gdi.getText("Interval"));
double scale = _wtof(gdi.getText("ScaleFactor").c_str());
bool reverse = bi.getExtraInt() == 2;
int pairSize = gdi.getSelectedItem("PairSize").first;
pSavedDepth = maxAfter;
pFirstRestart = deltaRestart;
pTimeScaling = scale;
pInterval = interval;
oListParam par;
for (int k = 0; k < oe->getNumClasses(); k++) {
if (!gdi.hasField("PLUse" + itos(k)))
continue;
BaseInfo *biu = gdi.setText("PLUse" + itos(k), L"", false);
if (biu) {
int id = biu->getExtraInt();
bool checked = gdi.isChecked("PLUse" + itos(k));
int first = oe->getRelativeTime(gdi.getText("First" + itos(k)));
map<int, PursuitSettings>::iterator st = pSettings.find(id);
if (st != pSettings.end()) {
st->second.firstTime = first;
st->second.maxTime = maxAfter;
st->second.use = checked;
}
if (checked) {
pClass pc = oe->getClass(id);
if (pc)
pc->setDrawFirstStart(first);
}
if (!cancel && checked) {
oe->drawPersuitList(id, first, first + deltaRestart, maxAfter,
interval, pairSize, reverse, scale);
par.selection.insert(id);
}
}
}
if (bi.id == "SavePursuit") {
return 0;
}
if (cancel) {
loadPage(gdi);
return 0;
}
gdi.restore("Pursuit", false);
gdi.dropLine();
gdi.fillDown();
oListInfo info;
par.listCode = EStdStartList;
oe->generateListInfo(par, gdi.getLineHeight(), info);
oe->generateList(gdi, false, info, true);
gdi.dropLine();
gdi.addButton("Cancel", "Återgå", ClassesCB);
gdi.refresh();
}
else if (bi.id.substr(0,5) == "PLUse") {
int k = atoi(bi.id.substr(5).c_str());
gdi.setInputStatus("First" + itos(k), gdi.isChecked(bi.id));
}
else if (bi.id == "AutomaticDraw") {
wstring firstStart = gdi.getText("FirstStart");
if (warnDrawStartTime(gdi, firstStart))
return 0;
wstring minInterval = gdi.getText("MinInterval");
wstring vacances = gdi.getText("Vacances");
bool lateBefore = false;
//bool pairwise = gdi.isChecked("Pairwise");
int pairSize = 1;
if (gdi.hasField("PairSize")) {
pairSize = gdi.getSelectedItem("PairSize").first;
}
int method = gdi.getSelectedItem("Method").first;
bool useSoft = method == DMSOFT;
clearPage(gdi, true);
oe->automaticDrawAll(gdi, firstStart, minInterval, vacances,
lateBefore, useSoft, pairSize);
oe->addAutoBib();
gdi.scrollToBottom();
gdi.addButton("Cancel", "Återgå", ClassesCB);
}
else if (bi.id == "SelectMisses") {
set<int> lst;
oe->getNotDrawnClasses(lst, true);
gdi.setSelection("Classes", lst);
enableLoadSettings(gdi);
}
else if (bi.id == "SelectNone") {
gdi.setSelection("Classes", set<int>());
enableLoadSettings(gdi);
}
else if (bi.id == "Simultaneous") {
wstring firstStart;
firstStart = gdi.getText("FirstStart");
clearPage(gdi, false);
gdi.addString("", boldLarge, "Gemensam start");
gdi.dropLine();
int by = 0;
int bx = gdi.getCX();
showClassSelection(gdi, bx, by, 0);
gdi.pushX();
gdi.addInput("FirstStart", firstStart, 10, 0, L"Starttid:");
gdi.dropLine(4);
gdi.popX();
gdi.fillRight();
gdi.addButton("AssignStart", "Tilldela", ClassesCB).isEdit(true);
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.addButton("EraseStartAll", "Radera starttider...", ClassesCB).isEdit(true).setExtra(1);
gdi.refresh();
}
else if (bi.id == "AssignStart") {
set<int> classes;
gdi.getSelection("Classes", classes);
if (classes.empty()) {
gdi.alert("Ingen klass vald.");
return 0;
}
wstring time = gdi.getText("FirstStart");
if (warnDrawStartTime(gdi, time))
return 0;
for (set<int>::iterator it = classes.begin(); it!=classes.end();++it) {
simultaneous(*it, time);
}
bi.id = "Simultaneous";
classCB(gdi, type, &bi);
}
else if (bi.id == "DrawAll") {
int origin = bi.getExtraInt();
wstring firstStart = oe->getAbsTime(3600);
wstring minInterval = L"2:00";
wstring vacances = L"5%";
int maxNumControl = 1;
int pairSize = 1;
//bool pairwise = false;
int by = 0;
int bx = gdi.getCX();
if (origin!=13) {
if (origin!=1) {
save(gdi, true);
ClassId = 0;
EditChanged=false;
}
else {
firstStart = gdi.getText("FirstStart");
minInterval = gdi.getText("MinInterval");
vacances = gdi.getText("Vacances");
//pairwise = gdi.isChecked("Pairwise");
if (gdi.hasField("PairSize")) {
pairSize = gdi.getSelectedItem("PairSize").first;
}
}
clearPage(gdi, false);
gdi.addString("", boldLarge, "Lotta flera klasser");
gdi.dropLine(0.5);
showClassSelection(gdi, bx, by, DrawClassesCB);
vector<pClass> cls;
oe->getClasses(cls, false);
set<int> clsId;
for (size_t k = 0; k < cls.size(); k++) {
if (cls[k]->hasFreeStart())
continue;
if (cls[k]->getStartType(0) != STDrawn)
continue;
clsId.insert(cls[k]->getId());
}
gdi.setSelection("Classes", clsId);
gdi.addString("", 1, "Grundinställningar");
gdi.pushX();
gdi.fillRight();
gdi.addInput("FirstStart", firstStart, 10, 0, L"Första start:");
gdi.addInput("nFields", L"10", 10, 0, L"Max parallellt startande:");
gdi.popX();
gdi.dropLine(3);
gdi.addSelection("MaxCommonControl", 150, 100, 0,
L"Max antal gemensamma kontroller:");
vector< pair<wstring, size_t> > items;
items.push_back(make_pair(lang.tl("Inga"), 1));
items.push_back(make_pair(lang.tl("Första kontrollen"), 2));
for (int k = 2; k<10; k++)
items.push_back(make_pair(lang.tl("X kontroller#" + itos(k)), k+1));
items.push_back(make_pair(lang.tl("Hela banan"), 1000));
gdi.addItem("MaxCommonControl", items);
gdi.selectItemByData("MaxCommonControl", maxNumControl);
gdi.popX();
gdi.dropLine(4);
gdi.addCheckbox("AllowNeighbours", "Tillåt samma bana inom basintervall", 0, true);
gdi.addCheckbox("CoursesTogether", "Lotta klasser med samma bana gemensamt", 0, false);
gdi.popX();
gdi.dropLine(2);
gdi.addString("", 1, "Startintervall");
gdi.dropLine(1.4);
gdi.popX();
gdi.fillRight();
gdi.addInput("BaseInterval", L"1:00", 10, 0, L"Basintervall (min):");
gdi.addInput("MinInterval", minInterval, 10, 0, L"Minsta intervall i klass:");
gdi.addInput("MaxInterval", minInterval, 10, 0, L"Största intervall i klass:");
gdi.popX();
gdi.dropLine(4);
gdi.addString("", 1, "Vakanser och efteranmälda");
gdi.dropLine(1.4);
gdi.popX();
gdi.addInput("Vacances", vacances, 10, 0, L"Andel vakanser:");
gdi.addInput("VacancesMin", L"1", 10, 0, L"Min. vakanser (per klass):");
gdi.addInput("VacancesMax", L"10", 10, 0, L"Max. vakanser (per klass):");
gdi.popX();
gdi.dropLine(3);
gdi.addInput("Extra", L"0%", 10, 0, L"Förväntad andel efteranmälda:");
gdi.dropLine(4);
gdi.fillDown();
gdi.popX();
gdi.setRestorePoint("Setup");
}
else {
gdi.restore("Setup");
by = gdi.getHeight();
gdi.enableEditControls(true);
}
gdi.fillRight();
gdi.pushX();
RECT rcPrepare;
rcPrepare.left = gdi.getCX();
rcPrepare.top = gdi.getCY();
gdi.setCX(gdi.getCX() + gdi.getLineHeight());
gdi.dropLine();
gdi.addString("", fontMediumPlus, "Förbered lottning");
gdi.dropLine(2.5);
gdi.popX();
gdi.setCX(gdi.getCX() + gdi.getLineHeight());
gdi.addButton("PrepareDrawAll", "Fördela starttider...", ClassesCB).isEdit(true).setDefault();
gdi.addButton("EraseStartAll", "Radera starttider...", ClassesCB).isEdit(true).setExtra(0);
gdi.addButton("LoadSettings", "Hämta inställningar från föregående lottning", ClassesCB).isEdit(true);
enableLoadSettings(gdi);
gdi.dropLine(3);
rcPrepare.bottom = gdi.getCY();
rcPrepare.right = gdi.getWidth();
gdi.addRectangle(rcPrepare, colorLightGreen);
gdi.dropLine();
gdi.popX();
gdi.addString("", 1, "Efteranmälningar");
gdi.dropLine(1.5);
gdi.popX();
gdi.addButton("DrawAllBefore", "Efteranmälda (före ordinarie)", ClassesCB).isEdit(true);
gdi.addButton("DrawAllAfter", "Efteranmälda (efter ordinarie)", ClassesCB).isEdit(true);
gdi.dropLine(4);
gdi.popX();
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.addButton("HelpDraw", "Hjälp...", ClassesCB, "");
gdi.dropLine(3);
by = max(by, gdi.getCY());
gdi.setCX(bx);
gdi.setCY(by);
gdi.fillDown();
gdi.dropLine();
gdi.setRestorePoint("ReadyToDistribute");
gdi.refresh();
}
else if (bi.id == "HelpDraw") {
gdioutput *gdi_new = getExtraWindow("help", true);
if (!gdi_new)
gdi_new = createExtraWindow("help", makeDash(L"MeOS - " + lang.tl(L"Hjälp")), gdi.scaleLength(640));
gdi_new->clearPage(true);
gdi_new->addString("", boldLarge, "Lotta flera klasser");
gdi_new->addString("", 10, "help_draw");
gdi_new->dropLine();
gdi_new->addButton("CloseWindow", "Stäng", ClassesCB);
}
else if (bi.id == "CloseWindow") {
gdi.closeWindow();
}
else if (bi.id=="PrepareDrawAll") {
set<int> classes;
gdi.getSelection("Classes", classes);
if (classes.empty()) {
throw meosException("Ingen klass vald.");
}
gdi.restore("ReadyToDistribute");
drawInfo.classes.clear();
for (set<int>::iterator it = classes.begin(); it!=classes.end();++it) {
map<int, ClassInfo>::iterator res = cInfoCache.find(*it);
if ( res != cInfoCache.end() ) {
res->second.hasFixedTime = false;
drawInfo.classes[*it] = res->second;
}
else
drawInfo.classes[*it] = ClassInfo(oe->getClass(*it));
}
readDrawInfo(gdi, drawInfo);
if (drawInfo.baseInterval <= 0 || drawInfo.baseInterval == NOTIME)
throw meosException("Ogiltigt basintervall.");
if (drawInfo.minClassInterval <= 0 || drawInfo.minClassInterval == NOTIME)
throw meosException("Ogiltigt minimalt intervall.");
if (drawInfo.minClassInterval > drawInfo.maxClassInterval || drawInfo.maxClassInterval == NOTIME)
throw meosException("Ogiltigt maximalt intervall. ");
if (drawInfo.minClassInterval < drawInfo.baseInterval) {
throw meosException("Startintervallet får inte vara kortare än basintervallet.");
}
if (drawInfo.minClassInterval % drawInfo.baseInterval != 0 ||
drawInfo.maxClassInterval % drawInfo.baseInterval != 0) {
throw meosException("Ett startintervall måste vara en multipel av basintervallet.");
}
gdi.enableEditControls(false);
if (drawInfo.firstStart<=0) {
drawInfo.baseInterval = 0;
gdi.addString("", 0, "Raderar starttider...");
}
oe->optimizeStartOrder(gdi, drawInfo, cInfo);
showClassSettings(gdi);
}
else if (bi.id == "LoadSettings") {
set<int> classes;
gdi.getSelection("Classes", classes);
if (classes.empty()) {
throw meosException("Ingen klass vald.");
}
gdi.restore("ReadyToDistribute");
oe->loadDrawSettings(classes, drawInfo, cInfo);
/*
drawInfo.firstStart = 3600 * 22;
drawInfo.minClassInterval = 3600;
drawInfo.maxClassInterval = 1;
drawInfo.minVacancy = 10;
set<int> reducedStart;
for (set<int>::iterator it = classes.begin(); it != classes.end(); ++it) {
pClass pc = oe->getClass(*it);
if (pc) {
int fs = pc->getDrawFirstStart();
int iv = pc->getDrawInterval();
if (iv > 0 && fs > 0) {
drawInfo.firstStart = min(drawInfo.firstStart, fs);
drawInfo.minClassInterval = min(drawInfo.minClassInterval, iv);
drawInfo.maxClassInterval = max(drawInfo.maxClassInterval, iv);
drawInfo.minVacancy = min(drawInfo.minVacancy, pc->getDrawVacant());
drawInfo.maxVacancy = max(drawInfo.maxVacancy, pc->getDrawVacant());
reducedStart.insert(fs%iv);
}
}
}
drawInfo.baseInterval = drawInfo.minClassInterval;
int lastStart = -1;
for (set<int>::iterator it = reducedStart.begin(); it != reducedStart.end(); ++it) {
if (lastStart == -1)
lastStart = *it;
else {
drawInfo.baseInterval = min(drawInfo.baseInterval, *it-lastStart);
lastStart = *it;
}
}
cInfo.clear();
cInfo.resize(classes.size());
int i = 0;
for (set<int>::iterator it = classes.begin(); it != classes.end(); ++it) {
pClass pc = oe->getClass(*it);
if (pc) {
int fs = pc->getDrawFirstStart();
int iv = pc->getDrawInterval();
cInfo[i].pc = pc;
cInfo[i].classId = *it;
cInfo[i].courseId = pc->getCourseId();
cInfo[i].firstStart = fs;
cInfo[i].unique = pc->getCourseId();
if (cInfo[i].unique == 0)
cInfo[i].unique = pc->getId() * 10000;
cInfo[i].firstStart = (fs - drawInfo.firstStart) / drawInfo.baseInterval;
cInfo[i].interval = iv / drawInfo.baseInterval;
cInfo[i].nVacant = pc->getDrawVacant();
i++;
}
}
*/
writeDrawInfo(gdi, drawInfo);
gdi.enableEditControls(false);
showClassSettings(gdi);
}
else if (bi.id == "VisualizeDraw") {
readClassSettings(gdi);
gdioutput *gdi_new = getExtraWindow(visualDrawWindow, true);
if (!gdi_new)
gdi_new = createExtraWindow(visualDrawWindow, makeDash(L"MeOS - " + lang.tl(L"Visualisera startfältet")), gdi.scaleLength(1000));
gdi_new->clearPage(false);
gdi_new->addString("", boldLarge, "Visualisera startfältet");
gdi_new->dropLine();
gdi_new->addString("", 0, "För muspekaren över en markering för att få mer information.");
gdi_new->dropLine();
visualizeField(*gdi_new);
gdi_new->dropLine();
gdi_new->addButton("CloseWindow", "Stäng", ClassesCB);
gdi_new->registerEvent("CloseWindow", 0).setHandler(&handleCloseWindow);
gdi_new->refresh();
}
else if (bi.id == "EraseStartAll") {
set<int> classes;
gdi.getSelection("Classes", classes);
if (classes.empty()) {
gdi.alert("Ingen klass vald.");
return 0;
}
if (classes.size() == 1) {
pClass pc = oe->getClass(*classes.begin());
if (!pc || !gdi.ask(L"Vill du verkligen radera alla starttider i X?#" + pc->getName()))
return 0;
}
else {
if (!gdi.ask(L"Vill du verkligen radera starttider i X klasser?#" + itow(classes.size())) )
return 0;
}
for (set<int>::const_iterator it = classes.begin(); it != classes.end(); ++it) {
vector<ClassDrawSpecification> spec;
spec.push_back(ClassDrawSpecification(*it, 0, 0, 0, 0));
oe->drawList(spec, false, 1, oEvent::drawAll);
}
if (bi.getExtraInt() == 1)
bi.id = "Simultaneous";
else
bi.id = "DrawAll";
bi.setExtra(1);
classCB(gdi, type, &bi); // Reload draw dialog
}
else if (bi.id == "DrawAdjust") {
readClassSettings(gdi);
gdi.restore("ReadyToDistribute");
oe->optimizeStartOrder(gdi, drawInfo, cInfo);
showClassSettings(gdi);
}
else if (bi.id == "DrawAllAdjust") {
readClassSettings(gdi);
bi.id = "DrawAll";
return classCB(gdi, type, &bi);
}
else if (bi.id == "DrawAllBefore" || bi.id == "DrawAllAfter") {
oe->drawRemaining(true, bi.id == "DrawAllAfter");
oe->addAutoBib();
loadPage(gdi);
}
else if (bi.id=="DoDraw" || bi.id=="DoDrawAfter" || bi.id=="DoDrawBefore"){
if (!checkClassSelected(gdi))
return false;
DWORD cid=ClassId;
DrawMethod method = DrawMethod(gdi.getSelectedItem("Method").first);
int interval = 0;
if (gdi.hasField("Interval"))
interval = convertAbsoluteTimeMS(gdi.getText("Interval"));
int vacanses = 0;
if (gdi.hasField("Vacanses"))
vacanses = gdi.getTextNo("Vacanses");
int leg = 0;
if (gdi.hasField("Leg")) {
leg = gdi.getSelectedItem("Leg").first;
}
wstring bib;
bool doBibs = false;
if (gdi.hasField("Bib")) {
bib = gdi.getText("Bib");
doBibs = gdi.isChecked("HandleBibs");
}
wstring time = gdi.getText("FirstStart");
int t=oe->getRelativeTime(time);
if (t<=0)
throw std::exception("Ogiltig första starttid. Måste vara efter nolltid.");
oEvent::DrawType dtype(oEvent::drawAll);
if (bi.id=="DoDrawAfter")
dtype = oEvent::remainingAfter;
else if (bi.id=="DoDrawBefore")
dtype = oEvent::remainingBefore;
else {
if (warnDrawStartTime(gdi, t))
return 0;
}
//bool pairwise = false;
// if (gdi.hasField("Pairwise"))
// pairwise = gdi.isChecked("Pairwise");
int pairSize = 1;
if (gdi.hasField("PairSize")) {
pairSize = gdi.getSelectedItem("PairSize").first;
}
int maxTime = 0, restartTime = 0;
double scaleFactor = 1.0;
if (gdi.hasField("TimeRestart"))
restartTime = oe->getRelativeTime(gdi.getText("TimeRestart"));
if (gdi.hasField("MaxAfter"))
maxTime = convertAbsoluteTimeMS(gdi.getText("MaxAfter"));
if (gdi.hasField("ScaleFactor"))
scaleFactor = _wtof(gdi.getText("ScaleFactor").c_str());
if (method == DMRandom || method == DMSOFT) {
vector<ClassDrawSpecification> spec;
spec.push_back(ClassDrawSpecification(cid, leg, t, interval, vacanses));
oe->drawList(spec, method == DMSOFT, pairSize, dtype);
}
else if (method == DMClumped)
oe->drawListClumped(cid, t, interval, vacanses);
else if (method == DMPursuit || method == DMReversePursuit) {
oe->drawPersuitList(cid, t, restartTime, maxTime,
interval, pairSize,
method == DMReversePursuit,
scaleFactor);
}
else if (method == DMSimultaneous) {
simultaneous(cid, time);
}
else if (method == DMSeeded) {
ListBoxInfo seedMethod;
gdi.getSelectedItem("SeedMethod", seedMethod);
wstring seedGroups = gdi.getText("SeedGroups");
vector<wstring> out;
split(seedGroups, L" ,;", out);
vector<int> sg;
bool invalid = false;
for (size_t k = 0; k < out.size(); k++) {
if (trim(out[k]).empty())
continue;
int val = _wtoi(trim(out[k]).c_str());
if (val <= 0)
invalid = true;
sg.push_back(val);
}
if (invalid || sg.empty())
throw meosException(L"Ogiltig storlek på seedningsgrupper X.#" + seedGroups);
bool noClubNb = gdi.isChecked("PreventClubNb");
bool reverse = gdi.isChecked("ReverseSeedning");
pClass pc=oe->getClass(ClassId);
if (!pc)
throw meosException("Class not found");
pc->drawSeeded(ClassSeedMethod(seedMethod.data), leg, t, interval,
sg, noClubNb, reverse, pairSize);
}
else
throw std::exception("Not implemented");
if (doBibs)
oe->addBib(cid, leg, bib);
// Clear input
gdi.restore("", false);
gdi.addButton("Cancel", "Återgå", ClassesCB).setCancel();
gdi.dropLine();
oListParam par;
par.selection.insert(cid);
oListInfo info;
par.listCode = EStdStartList;
par.setLegNumberCoded(leg);
oe->generateListInfo(par, gdi.getLineHeight(), info);
oe->generateList(gdi, false, info, true);
gdi.refresh();
gdi.setData("ClassPageLoaded", 1);
return 0;
}
else if (bi.id=="HandleBibs") {
gdi.setInputStatus("Bib", gdi.isChecked("HandleBibs"));
}
else if (bi.id == "DoDeleteStart") {
pClass pc=oe->getClass(ClassId);
if (!pc)
throw meosException("Class not found");
if (!gdi.ask(L"Vill du verkligen radera alla starttider i X?#" + pc->getName()))
return 0;
int leg = 0;
if (gdi.hasField("Leg")) {
leg = gdi.getSelectedItem("Leg").first;
}
vector<ClassDrawSpecification> spec;
spec.push_back(ClassDrawSpecification(ClassId, leg, 0, 0, 0));
oe->drawList(spec, false, 1, oEvent::drawAll);
loadPage(gdi);
}
else if (bi.id=="Draw") {
save(gdi, true);
if (!checkClassSelected(gdi))
return false;
DWORD cid=ClassId;
if (oe->classHasResults(cid)) {
if (!gdi.ask(L"warning:drawresult"))
return 0;
}
pClass pc=oe->getClass(cid);
if (!pc)
throw std::exception("Class not found");
if (EditChanged)
gdi.sendCtrlMessage("Save");
clearPage(gdi, false);
gdi.addString("", boldLarge, L"Lotta klassen X#"+pc->getName());
gdi.dropLine();
gdi.pushX();
gdi.setRestorePoint();
gdi.fillDown();
bool multiDay = oe->hasPrevStage();
if (multiDay) {
gdi.addCheckbox("HandleMultiDay", "Använd funktioner för fleretappsklass", ClassesCB, true);
}
gdi.addSelection("Method", 200, 200, ClassesCB, L"Metod:");
gdi.dropLine(1.5);
gdi.popX();
gdi.setRestorePoint("MultiDayDraw");
lastDrawMethod = NOMethod;
drawDialog(gdi, getDefaultMethod(true), *pc);
}
else if (bi.id == "HandleMultiDay") {
ListBoxInfo lbi;
gdi.getSelectedItem("Method", lbi);
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Class not found");
if (gdi.isChecked(bi.id) && (lastDrawMethod == DMReversePursuit ||
lastDrawMethod == DMPursuit)) {
drawDialog(gdi, getDefaultMethod(false), *pc);
}
else
setMultiDayClass(gdi, gdi.isChecked(bi.id), lastDrawMethod);
}
else if (bi.id=="Bibs") {
save(gdi, true);
if (!checkClassSelected(gdi))
return false;
DWORD cid=ClassId;
pClass pc=oe->getClass(cid);
if (!pc)
throw std::exception("Class not found");
clearPage(gdi, false);
gdi.addString("", boldLarge, L"Nummerlappar i X#" + pc->getName());
gdi.dropLine();
gdi.setRestorePoint("bib");
gdi.addString("", 10, "help:bibs");
gdi.dropLine();
vector< pair<wstring, size_t> > bibOptions;
vector< pair<wstring, size_t> > bibTeamOptions;
bibOptions.push_back(make_pair(lang.tl("Manuell"), AutoBibManual));
bibOptions.push_back(make_pair(lang.tl("Löpande"), AutoBibConsecutive));
bibOptions.push_back(make_pair(lang.tl("Ingen"), AutoBibNone));
bibOptions.push_back(make_pair(lang.tl("Automatisk"), AutoBibExplicit));
gdi.fillRight();
gdi.pushX();
gdi.addSelection("BibSettings", 150, 100, ClassesCB, L"Metod:");
gdi.addItem("BibSettings", bibOptions);
AutoBibType bt = pc->getAutoBibType();
gdi.selectItemByData("BibSettings", bt);
wstring bib = pc->getDCI().getString("Bib");
if (pc->getNumDistinctRunners() > 1) {
bibTeamOptions.push_back(make_pair(lang.tl("Oberoende"), BibFree));
bibTeamOptions.push_back(make_pair(lang.tl("Samma"), BibSame));
bibTeamOptions.push_back(make_pair(lang.tl("Ökande"), BibAdd));
bibTeamOptions.push_back(make_pair(lang.tl("Sträcka"), BibLeg));
gdi.addSelection("BibTeam", 80, 100, 0, L"Lagmedlem:", L"Ange relation mellan lagets och deltagarnas nummerlappar.");
gdi.addItem("BibTeam", bibTeamOptions);
gdi.selectItemByData("BibTeam", pc->getBibMode());
}
gdi.dropLine(1.1);
gdi.addInput("Bib", L"", 10, 0, L"");
gdi.dropLine(3);
gdi.fillRight();
gdi.popX();
gdi.addString("", 0, "Antal reserverade nummerlappsnummer mellan klasser:");
gdi.dropLine(-0.1);
gdi.addInput("BibGap", itow(oe->getBibClassGap()), 5);
gdi.dropLine(3);
gdi.popX();
gdi.fillRight();
gdi.addButton("DoBibs", "Tilldela", ClassesCB).setDefault();
gdi.setInputStatus("Bib", bt == AutoBibExplicit || bt == AutoBibManual);
gdi.setInputStatus("BibGap", bt != AutoBibManual);
if (bt != AutoBibManual)
gdi.setTextTranslate("DoBibs", L"OK");
if (bt == AutoBibExplicit)
gdi.setText("Bib", bib);
gdi.fillDown();
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.popX();
EditChanged=false;
gdi.refresh();
}
else if (bi.id=="DoBibs") {
if (!checkClassSelected(gdi))
return false;
DWORD cid=ClassId;
pClass pc = oe->getClass(cid);
AutoBibType bt = AutoBibType(gdi.getSelectedItem("BibSettings").first);
pair<int, bool> teamBib = gdi.getSelectedItem("TeamBib");
if (teamBib.second) {
pc->setBibMode(BibMode(teamBib.first));
}
pc->getDI().setString("Bib", getBibCode(bt, gdi.getText("Bib")));
pc->synchronize();
if (bt == AutoBibManual) {
oe->addBib(cid, 0, gdi.getText("Bib"));
}
else {
oe->setBibClassGap(gdi.getTextNo("BibGap"));
oe->addAutoBib();
}
gdi.restore("bib", false);
gdi.dropLine();
gdi.addButton("Cancel", "Återgå", ClassesCB).setDefault();
oListParam par;
par.selection.insert(cid);
oListInfo info;
par.listCode = EStdStartList;
par.setLegNumberCoded(0);
oe->generateListInfo(par, gdi.getLineHeight(), info);
oe->generateList(gdi, false, info, true);
gdi.refresh();
return 0;
}
else if (bi.id=="Split") {
save(gdi, true);
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Class not found");
clearPage(gdi, true);
gdi.addString("", boldLarge, L"Dela klass: X#" + pc->getName());
gdi.dropLine();
int tot, fin, dns;
oe->getNumClassRunners(pc->getId(), 0, tot, fin, dns);
gdi.addString("", fontMediumPlus, "Antal deltagare: X#" + itos(tot));
gdi.dropLine(1.2);
gdi.pushX();
gdi.fillRight();
gdi.addSelection("Type", 200, 100, ClassesCB, L"Typ av delning:");
gdi.selectItemByData("Type", 1);
vector< pair<wstring, size_t> > mt;
oClass::getSplitMethods(mt);
gdi.addItem("Type", mt);
gdi.selectFirstItem("Type");
gdi.addSelection("SplitInput", 100, 150, ClassesCB, L"Antal klasser:").setExtra(tot);
vector< pair<wstring, size_t> > sp;
for (int k = 2; k < 10; k++)
sp.push_back(make_pair(itow(k), k));
gdi.addItem("SplitInput", sp);
gdi.selectFirstItem("SplitInput");
gdi.dropLine(3);
gdi.popX();
updateSplitDistribution(gdi, 2, tot);
}
else if (bi.id=="DoSplit") {
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Class not found");
ListBoxInfo lbi;
gdi.getSelectedItem("Type", lbi);
int number = gdi.getSelectedItem("SplitInput").first;
vector<int> parts(number);
for (int k = 0; k < number; k++) {
string id = "CLS" + itos(k);
parts[k] = gdi.getTextNo(id, false);
}
vector<int> outClass;
pc->splitClass(ClassSplitMethod(lbi.data), parts, outClass);
clearPage(gdi, true);
gdi.addButton("Cancel", "Återgå", ClassesCB);
oListParam par;
par.selection.insert(outClass.begin(), outClass.end());
oListInfo info;
par.listCode = EStdStartList;
oe->generateListInfo(par, gdi.getLineHeight(), info);
oe->generateList(gdi, false, info, true);
}
else if (bi.id == "LockAllForks" || bi.id == "UnLockAllForks") {
bool lock = bi.id == "LockAllForks";
vector<pClass> allCls;
oe->getClasses(allCls, true);
for (pClass c : allCls) {
if (c->isRemoved())
continue;
if (!c->hasCoursePool() && c->hasMultiCourse()) {
c->lockedForking(lock);
c->synchronize(true);
}
}
loadPage(gdi);
return 0;
}
else if (bi.id=="Merge") {
save(gdi, true);
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Class not found");
vector< pair<wstring, size_t> > rawClass, cls;
oe->fillClasses(rawClass, oEvent::extraNone, oEvent::filterNone);
int def = -1;
bool next = false;
for (size_t k = 0; k < rawClass.size(); k++) {
if (rawClass[k].second == ClassId)
next = true;
else {
cls.push_back(rawClass[k]);
if (next) {
next = false;
def = rawClass[k].second;
}
}
}
if (cls.empty())
throw std::exception("En klass kan inte slås ihop med sig själv.");
clearPage(gdi, true);
gdi.addString("", boldLarge, L"Slå ihop klass: X (denna klass behålls)#" + pc->getName());
gdi.dropLine();
gdi.addString("", 10, "help:12138");
gdi.dropLine(2);
gdi.pushX();
gdi.fillRight();
gdi.addSelection("Class", 150, 300, 0, L"Klass att slå ihop:");
gdi.addItem("Class", cls);
if (def != -1)
gdi.selectItemByData("Class", def);
else
gdi.selectFirstItem("Class");
gdi.dropLine();
gdi.addButton("DoMergeAsk", "Slå ihop", ClassesCB).setDefault();
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.dropLine(3);
gdi.popX();
}
else if (bi.id=="DoMergeAsk") {
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Class not found");
pClass mergeClass = oe->getClass(gdi.getSelectedItem("Class").first);
if (!mergeClass)
throw std::exception("Ingen klass vald.");
if (mergeClass->getId() == ClassId)
throw std::exception("En klass kan inte slås ihop med sig själv.");
if (gdi.ask(L"Vill du flytta löpare från X till Y och ta bort Z?#"
+ mergeClass->getName() + L"#" + pc->getName() + L"#" + mergeClass->getName())) {
bi.id = "DoMerge";
return classCB(gdi, type, &bi);
}
return false;
}
else if (bi.id=="DoMerge") {
if (!checkClassSelected(gdi))
return false;
pClass pc=oe->getClass(ClassId);
if (!pc)
throw std::exception("Class not found");
ListBoxInfo lbi;
gdi.getSelectedItem("Class", lbi);
if (signed(lbi.data)<=0)
throw std::exception("Ingen klass vald.");
if (lbi.data==ClassId)
throw std::exception("En klass kan inte slås ihop med sig själv.");
pc->mergeClass(lbi.data);
clearPage(gdi, true);
gdi.addButton("Cancel", "Återgå", ClassesCB);
oListParam par;
par.selection.insert(ClassId);
oListInfo info;
par.listCode = EStdStartList;
oe->generateListInfo(par, gdi.getLineHeight(), info);
oe->generateList(gdi, false, info, true);
gdi.refresh();
}
else if (bi.id=="MultiCourse") {
save(gdi, false);
multiCourse(gdi, 0);
gdi.refresh();
}
else if (bi.id=="Save")
save(gdi, false);
else if (bi.id=="Add") {
wstring name = gdi.getText("Name");
pClass c = oe->getClass(ClassId);
if (!name.empty() && c && c->getName() != name) {
if (gdi.ask(L"Vill du lägga till klassen 'X'?#" + name)) {
c = oe->addClass(name);
ClassId = c->getId();
save(gdi, false);
return true;
}
}
save(gdi, true);
pClass pc = oe->addClass(oe->getAutoClassName(), 0);
if (pc) {
oe->fillClasses(gdi, "Classes", oEvent::extraDrawn, oEvent::filterNone);
selectClass(gdi, pc->getId());
gdi.setInputFocus("Name", true);
}
}
else if (bi.id=="Remove") {
EditChanged=false;
if (!checkClassSelected(gdi))
return false;
DWORD cid=ClassId;
if (oe->isClassUsed(cid))
gdi.alert("Klassen används och kan inte tas bort.");
else
oe->removeClass(cid);
oe->fillClasses(gdi, "Classes", oEvent::extraDrawn, oEvent::filterNone);
ClassId = 0;
selectClass(gdi, 0);
}
}
else if (type==GUI_LISTBOX) {
ListBoxInfo bi=*(ListBoxInfo *)data;
if (bi.id=="Classes") {
if (gdi.isInputChanged(""))
save(gdi, true);
selectClass(gdi, bi.data);
}
else if (bi.id == "SplitInput") {
int num = bi.data;
updateSplitDistribution(gdi, num, bi.getExtraInt());
}
else if (bi.id=="Courses")
EditChanged=true;
else if (bi.id == "BibSettings") {
AutoBibType bt = (AutoBibType)bi.data;
gdi.setInputStatus("Bib", bt == AutoBibExplicit || bt == AutoBibManual);
gdi.setInputStatus("BibGap", bt != AutoBibManual);
if (bt != AutoBibManual)
gdi.setTextTranslate("DoBibs", L"OK");
else
gdi.setTextTranslate("DoBibs", L"Tilldela");
}
else if (bi.id=="Type") {
if (bi.data==1) {
gdi.setTextTranslate("TypeDesc", L"Antal klasser:", true);
gdi.setText("SplitInput", L"2");
}
else {
gdi.setTextTranslate("TypeDesc", L"Löpare per klass:", true);
gdi.setText("SplitInput", L"100");
}
}
else if (bi.id == "Method") {
pClass pc = oe->getClass(ClassId);
if (!pc)
throw std::exception("Nullpointer exception");
drawDialog(gdi, DrawMethod(bi.data), *pc);
}
}
else if (type==GUI_INPUTCHANGE) {
//InputInfo ii=*(InputInfo *)data;
}
else if (type==GUI_CLEAR) {
if (ClassId>0)
save(gdi, true);
if (EditChanged) {
if (gdi.ask(L"Spara ändringar?"))
gdi.sendCtrlMessage("Save");
}
return true;
}
return 0;
}
void TabClass::readClassSettings(gdioutput &gdi)
{
for (size_t k=0;k<cInfo.size();k++) {
ClassInfo &ci = cInfo[k];
int id = ci.classId;
const wstring &start = gdi.getText("S"+itos(id));
const wstring &intervall = gdi.getText("I"+itos(id));
int vacant = _wtoi(gdi.getText("V"+itos(id)).c_str());
int reserved = _wtoi(gdi.getText("R"+itos(id)).c_str());
int startPos = oe->getRelativeTime(start) - drawInfo.firstStart;
if (drawInfo.firstStart == 0 && startPos == -1)
startPos = 0;
else if (startPos<0 || (startPos % drawInfo.baseInterval)!=0)
throw std::exception("Ogiltig tid");
startPos /= drawInfo.baseInterval;
int intervalPos = convertAbsoluteTimeMS(intervall);
if (intervalPos<0 || intervalPos == NOTIME || (intervalPos % drawInfo.baseInterval)!=0)
throw std::exception("Ogiltigt startintervall");
intervalPos /= drawInfo.baseInterval;
if (ci.nVacant != vacant) {
ci.nVacantSpecified = true;
ci.nVacant = vacant;
}
if (ci.nExtra != reserved) {
ci.nExtraSpecified = true;
ci.nExtra = reserved;
}
// If times has been changed, mark this class to be kept fixed
if (ci.firstStart != startPos || ci.interval!=intervalPos)
ci.hasFixedTime = true;
ci.firstStart = startPos;
ci.interval = intervalPos;
drawInfo.classes[ci.classId] = ci;
cInfoCache[ci.classId] = ci;
cInfoCache[ci.classId].hasFixedTime = false;
}
}
void TabClass::visualizeField(gdioutput &gdi) {
ClassInfo::sSortOrder = 3;
sort(cInfo.begin(), cInfo.end());
ClassInfo::sSortOrder = 0;
vector<int> field;
vector<int> index(cInfo.size(), -1);
for (size_t k = 0;k < cInfo.size(); k++) {
const ClassInfo &ci = cInfo[k];
int laststart = ci.firstStart + (ci.nRunners-1) * ci.interval;
for (size_t j = 0; j < field.size(); j++) {
if (field[j] < ci.firstStart) {
index[k] = j;
field[j] = laststart;
break;
}
}
if (index[k] == -1) {
index[k] = field.size();
field.push_back(laststart);
}
/*
string first=oe->getAbsTime(ci.firstStart*drawInfo.baseInterval+drawInfo.firstStart);
string last=oe->getAbsTime((laststart)*drawInfo.baseInterval+drawInfo.firstStart);
pClass pc=oe->getClass(ci.classId);*/
}
map<int, int> groupNumber;
map<int, wstring> groups;
int freeNumber = 1;
for (size_t k = 0;k < cInfo.size(); k++) {
const ClassInfo &ci = cInfo[k];
if (!groupNumber.count(ci.unique))
groupNumber[ci.unique] = freeNumber++;
pClass pc = oe->getClass(ci.classId);
if (pc) {
if (groups[ci.unique].empty())
groups[ci.unique] = pc->getName();
else if (groups[ci.unique].size() < 64)
groups[ci.unique] += L", " + pc->getName();
else
groups[ci.unique] += L"...";
}
}
int marg = gdi.scaleLength(20);
int xp = gdi.getCX() + marg;
int yp = gdi.getCY() + marg;
int h = gdi.scaleLength(12);
int w = gdi.scaleLength(6);
int maxx = xp, maxy = yp;
RECT rc;
for (size_t k = 0;k < cInfo.size(); k++) {
const ClassInfo &ci = cInfo[k];
rc.top = yp + index[k] * h;
rc.bottom = rc.top + h - 1;
int g = ci.unique;
GDICOLOR color = GDICOLOR(RGB(((g * 30)&0xFF), ((g * 50)&0xFF), ((g * 70)&0xFF)));
for (int j = 0; j<ci.nRunners; j++) {
rc.left = xp + (ci.firstStart + j * ci.interval) * w;
rc.right = rc.left + w-1;
gdi.addRectangle(rc, color);
}
pClass pc = oe->getClass(ci.classId);
if (pc) {
wstring course = pc->getCourse() ? L", " + pc->getCourse()->getName() : L"";
wstring tip = L"X (Y deltagare, grupp Z, W)#" + pc->getName() + course + L"#" +
itow(ci.nRunners) + L"#" + itow(groupNumber[ci.unique])
+ L"#" + groups[ci.unique];
rc.left = xp + ci.firstStart * w;
int laststart = ci.firstStart + (ci.nRunners-1) * ci.interval;
rc.right = xp + (laststart + 1) * w;
gdi.addToolTip("", tip, 0, &rc);
maxx = max<int>(maxx, rc.right);
maxy = max<int>(maxy, rc.bottom);
}
}
rc.left = xp - marg;
rc.top = yp - marg;
rc.bottom = maxy + marg;
rc.right = maxx + marg;
gdi.addRectangle(rc, colorLightYellow, true, true);
}
void TabClass::showClassSettings(gdioutput &gdi)
{
ClassInfo::sSortOrder = 2;
sort(cInfo.begin(), cInfo.end());
ClassInfo::sSortOrder=0;
int laststart=0;
for (size_t k=0;k<cInfo.size();k++) {
const ClassInfo &ci = cInfo[k];
laststart=max(laststart, ci.firstStart+ci.nRunners*ci.interval);
}
int y = 0;
int xp = gdi.getCX();
const int width = 80;
if (!cInfo.empty()) {
gdi.dropLine();
y = gdi.getCY();
gdi.addString("", y, xp, 1, "Sammanställning, klasser:");
gdi.addString("", y, xp+300, 0, "Första start:");
gdi.addString("", y, xp+300+width, 0, "Intervall:");
gdi.addString("", y, xp+300+width*2, 0, "Vakanser:");
gdi.addString("", y, xp+300+width*3, 0, "Reserverade:");
}
gdi.pushX();
for (size_t k=0;k<cInfo.size();k++) {
const ClassInfo &ci = cInfo[k];
wchar_t bf1[128];
wchar_t bf2[128];
int cstart = ci.firstStart + (ci.nRunners-1) * ci.interval;
wstring first=oe->getAbsTime(ci.firstStart*drawInfo.baseInterval+drawInfo.firstStart);
wstring last=oe->getAbsTime((cstart)*drawInfo.baseInterval+drawInfo.firstStart);
pClass pc=oe->getClass(ci.classId);
swprintf_s(bf1, L"%s, %d", pc ? pc->getName().c_str() : L"-", ci.nRunners);
swprintf_s(bf2, L"%d-[%d]-%d (%s-%s)", ci.firstStart, ci.interval, cstart, first.c_str(), last.c_str());
gdi.fillRight();
int id = ci.classId;
gdi.addString("C" + itos(id), 0, L"X platser. Startar Y#" + wstring(bf1) + L"#" + bf2);
y = gdi.getCY();
gdi.addInput(xp+300, y, "S"+itos(id), first, 7, DrawClassesCB);
gdi.addInput(xp+300+width, y, "I"+itos(id), formatTime(ci.interval*drawInfo.baseInterval), 7, DrawClassesCB);
gdi.addInput(xp+300+width*2, y, "V"+itos(id), itow(ci.nVacant), 7, DrawClassesCB);
gdi.addInput(xp+300+width*3, y, "R"+itos(id), itow(ci.nExtra), 7, DrawClassesCB);
if (k%5 == 4)
gdi.dropLine(1);
gdi.dropLine(1.6);
gdi.fillDown();
gdi.popX();
}
gdi.dropLine();
gdi.pushX();
gdi.fillRight();
gdi.addButton("VisualizeDraw", "Visualisera startfältet...", ClassesCB);
gdi.addButton("SaveDrawSettings", "Spara starttider", ClassesCB,
"Spara inmatade tider i tävlingen utan att tilldela starttider.");
gdi.addButton("DrawAllAdjust", "Ändra inställningar", ClassesCB,
"Ändra grundläggande inställningar och gör en ny fördelning").setExtra(13);
if (!cInfo.empty()) {
gdi.addButton("DrawAdjust", "Uppdatera fördelning", ClassesCB,
"Uppdatera fördelningen av starttider med hänsyn till manuella ändringar ovan");
gdi.disableInput("DrawAdjust");
}
gdi.popX();
gdi.dropLine(3);
gdi.fillRight();
if (!cInfo.empty()) {
gdi.pushX();
RECT rc;
rc.left = gdi.getCX();
rc.top = gdi.getCY();
rc.bottom = rc.top + gdi.getButtonHeight() + gdi.scaleLength(22) + gdi.getLineHeight();
gdi.setCX(rc.left + gdi.scaleLength(10));
gdi.setCY(rc.top + gdi.scaleLength(10));
gdi.addSelection("Method", 200, 200, 0, L"Metod:");
gdi.addItem("Method", lang.tl("Lottning"), DMRandom);
gdi.addItem("Method", lang.tl("SOFT-lottning"), DMSOFT);
gdi.selectItemByData("Method", getDefaultMethod(false));
gdi.addSelection("PairSize", 150, 200, 0, L"Tillämpa parstart:");
gdi.addItem("PairSize", getPairOptions());
gdi.selectItemByData("PairSize", 1);
gdi.dropLine(0.9);
gdi.addButton("DoDrawAll", "Utför lottning", ClassesCB);
rc.right = gdi.getCX() + gdi.scaleLength(5);
gdi.addRectangle(rc, colorLightGreen);
gdi.setCX(rc.right + gdi.scaleLength(10));
}
gdi.addButton("Cancel", "Avbryt", ClassesCB);
gdi.fillDown();
gdi.dropLine(2);
gdi.popX();
gdi.fillDown();
gdi.popX();
gdi.scrollToBottom();
gdi.updateScrollbars();
gdi.refresh();
}
void TabClass::selectClass(gdioutput &gdi, int cid)
{
oe->fillCourses(gdi, "Courses", true);
gdi.addItem("Courses", lang.tl("Ingen bana"), -2);
if (cid==0) {
gdi.restore("", true);
gdi.disableInput("MultiCourse", true);
gdi.enableInput("Courses");
gdi.enableEditControls(false);
gdi.setText("Name", L"");
gdi.selectItemByData("Courses", -2);
gdi.check("AllowQuickEntry", true);
if (gdi.hasField("FreeStart"))
gdi.check("FreeStart", false);
if (gdi.hasField("IgnoreStart"))
gdi.check("IgnoreStart", false);
if (gdi.hasField("DirectResult"))
gdi.check("DirectResult", false);
gdi.check("NoTiming", false);
ClassId=cid;
EditChanged=false;
gdi.disableInput("Remove");
gdi.disableInput("Save");
return;
}
pClass pc = oe->getClass(cid);
if (!pc) {
selectClass(gdi, 0);
return;
}
gdi.enableEditControls(true);
gdi.enableInput("Remove");
gdi.enableInput("Save");
pc->synchronize();
gdi.setText("Name", pc->getName());
gdi.setText("ClassType", pc->getType());
gdi.setText("StartName", pc->getStart());
if (pc->getBlock()>0)
gdi.selectItemByData("StartBlock", pc->getBlock());
else
gdi.selectItemByData("StartBlock", -1);
if (gdi.hasField("Status")) {
vector< pair<wstring, size_t> > out;
size_t selected = 0;
pc->getDCI().fillInput("Status", out, selected);
gdi.addItem("Status", out);
gdi.selectItemByData("Status", selected);
}
gdi.check("AllowQuickEntry", pc->getAllowQuickEntry());
gdi.check("NoTiming", pc->getNoTiming());
if (gdi.hasField("FreeStart"))
gdi.check("FreeStart", pc->hasFreeStart());
if (gdi.hasField("IgnoreStart"))
gdi.check("IgnoreStart", pc->ignoreStartPunch());
if (gdi.hasField("DirectResult"))
gdi.check("DirectResult", pc->hasDirectResult());
ClassId=cid;
if (pc->hasTrueMultiCourse()) {
gdi.restore("", false);
multiCourse(gdi, pc->getNumStages());
gdi.refresh();
gdi.addItem("Courses", lang.tl("Flera banor"), -3);
gdi.selectItemByData("Courses", -3);
gdi.disableInput("Courses");
gdi.check("CoursePool", pc->hasCoursePool());
if (gdi.hasField("Unordered"))
gdi.check("Unordered", pc->hasUnorderedLegs());
if (gdi.hasField("LockForking")) {
gdi.check("LockForking", pc->lockedForking());
setLockForkingState(gdi, pc->hasCoursePool(), pc->lockedForking());
}
if (gdi.hasField("MCourses")) {
oe->fillCourses(gdi, "MCourses", true);
string strId = "StageCourses_label";
gdi.setTextTranslate(strId, getCourseLabel(pc->hasCoursePool()), true);
}
if (gdi.hasData("SimpleMulti")) {
bool hasStart = pc->getStartType(0) == STTime;
gdi.setInputStatus("CommonStartTime", hasStart);
gdi.check("CommonStart", hasStart);
if (hasStart)
gdi.setText("CommonStartTime", pc->getStartDataS(0));
else
gdi.setText("CommonStartTime", makeDash(L"-"));
}
else {
updateFairForking(gdi, pc);
int nstage=pc->getNumStages();
gdi.setText("NStage", nstage);
for (int k=0;k<nstage;k++) {
char legno[10];
sprintf_s(legno, "%d", k);
gdi.selectItemByData((string("LegType")+legno).c_str(), pc->getLegType(k));
gdi.selectItemByData((string("StartType")+legno).c_str(), pc->getStartType(k));
updateStartData(gdi, pc, k, false, true);
gdi.setInputStatus(string("Restart")+legno, !pc->restartIgnored(k), true);
gdi.setInputStatus(string("RestartRope")+legno, !pc->restartIgnored(k), true);
if (gdi.hasField(string("Restart")+legno))
gdi.setText(string("Restart")+legno, pc->getRestartTimeS(k));
if (gdi.hasField(string("RestartRope")+legno))
gdi.setText(string("RestartRope")+legno, pc->getRopeTimeS(k));
if (gdi.hasField(string("MultiR")+legno))
gdi.selectItemByData((string("MultiR")+legno).c_str(), pc->getLegRunner(k));
}
}
}
else {
gdi.restore("", true);
gdi.enableInput("MultiCourse", true);
gdi.enableInput("Courses");
pCourse pcourse = pc->getCourse();
gdi.selectItemByData("Courses", pcourse ? pcourse->getId():-2);
}
gdi.selectItemByData("Classes", cid);
ClassId=cid;
EditChanged=false;
}
void TabClass::legSetup(gdioutput &gdi)
{
gdi.restore("RelaySetup");
gdi.pushX();
gdi.fillDown();
gdi.addString("", 10, "help:relaysetup");
gdi.dropLine();
gdi.addSelection("Predefined", 150, 200, MultiCB, L"Fördefinierade tävlingsformer:").ignore(true);
oe->fillPredefinedCmp(gdi, "Predefined");
if (storedPredefined == oEvent::PredefinedTypes(-1)) {
bool hasPatrol = oe->getMeOSFeatures().hasFeature(MeOSFeatures::Patrol);
bool hasRelay = oe->getMeOSFeatures().hasFeature(MeOSFeatures::Relay);
if (hasRelay)
storedPredefined = oEvent::PRelay;
else if (hasPatrol)
storedPredefined = oEvent::PPatrol;
else
storedPredefined = oEvent::PNoSettings;
}
gdi.selectItemByData("Predefined", storedPredefined);
gdi.fillRight();
gdi.addInput("NStage", storedNStage, 4, MultiCB, L"Antal sträckor:").ignore(true);
gdi.addInput("StartTime", storedStart, 6, MultiCB, L"Starttid (HH:MM:SS):");
gdi.popX();
bool nleg;
bool start;
oe->setupRelayInfo(storedPredefined, nleg, start);
gdi.setInputStatus("NStage", nleg);
gdi.setInputStatus("StartTime", start);
gdi.fillRight();
gdi.dropLine(3);
gdi.addButton("SetNStage", "Verkställ", MultiCB);
gdi.fillDown();
gdi.addButton("Cancel", "Avbryt", ClassesCB);
gdi.popX();
}
void TabClass::multiCourse(gdioutput &gdi, int nLeg) {
currentStage=-1;
bool simpleView = nLeg==1;
bool showGuide = (oe->getMeOSFeatures().hasFeature(MeOSFeatures::Relay) ||
oe->getMeOSFeatures().hasFeature(MeOSFeatures::Patrol)) && nLeg==0;
if (nLeg == 0 && !showGuide) {
pClass pc = oe->getClass(ClassId);
if (pc) {
pc->setNumStages(1);
pc->setStartType(0, STDrawn, false);
pc->forceShowMultiDialog(true);
selectClass(gdi, ClassId);
return;
}
}
gdi.disableInput("MultiCourse", true);
gdi.setRestorePoint();
gdi.fillDown();
gdi.newColumn();
int cx=gdi.getCX(), cy=gdi.getCY();
gdi.setCX(cx+10);
gdi.setCY(cy+10);
if (simpleView) {
gdi.addString("", fontMediumPlus, "Gafflade banor");
}
else {
gdi.addString("", 2, "Flera banor / stafett / patrull / banpool");
gdi.addString("", 0, "Låt klassen ha mer än en bana eller sträcka");
gdi.dropLine();
}
gdi.setRestorePoint("RelaySetup");
if (showGuide) {
legSetup(gdi);
RECT rc;
rc.left = cx;
rc.right = gdi.getWidth()+10;
rc.bottom = gdi.getCY()+10;
rc.top = cy;
gdi.addRectangle(rc, colorLightGreen, true, false).set3D(true).setColor2(colorLightCyan);
}
else if (simpleView) {
gdi.fillRight();
gdi.pushX();
gdi.setData("SimpleMulti", 1);
gdi.dropLine();
gdi.addCheckbox("CommonStart", "Gemensam start", MultiCB, false);
//gdi.dropLine(-1);
gdi.addInput("CommonStartTime", L"", 10, 0, L"");
gdi.fillDown();
gdi.popX();
gdi.dropLine(2);
gdi.addCheckbox("CoursePool", "Använd banpool", MultiCB, false,
"Knyt löparna till banor från en pool vid målgång.");
gdi.addButton("OneCourse", "Endast en bana", MultiCB, "Använd endast en bana i klassen");
gdi.setRestorePoint("Courses");
selectCourses(gdi, 0);
RECT rc;
rc.left = cx;
rc.right = gdi.getWidth()+10;
rc.bottom = gdi.getCY()+10;
rc.top = cy;
gdi.addRectangle(rc, colorLightBlue, true, false).set3D(true);
}
else {
gdi.pushX();
gdi.fillRight();
gdi.addButton("ChangeLeg", "Ändra klassinställningar...", MultiCB, "Starta en guide som hjälper dig göra klassinställningar");
gdi.fillDown();
gdi.popX();
gdi.dropLine(2);
gdi.dropLine(0.5);
int headYPos=gdi.getCY();
gdi.dropLine(1.2);
vector< pair<wstring, size_t> > legs;
legs.reserve(nLeg);
for (int j=0;j<nLeg;j++) {
wchar_t bf[16];
swprintf_s(bf, lang.tl("Str. %d").c_str(), j+1);
legs.push_back( make_pair(bf, j) );
}
bool multipleRaces = oe->getMeOSFeatures().hasFeature(MeOSFeatures::MultipleRaces);
bool hasRelay = oe->getMeOSFeatures().hasFeature(MeOSFeatures::Relay);
for (int k=0;k<nLeg;k++) {
char legno[10];
int headXPos[10];
sprintf_s(legno, "%d", k);
gdi.fillRight();
headXPos[0]=gdi.getCX();
gdi.addStringUT(gdi.getCY(), gdi.getCX(), 1, itos(k+1) + ".");
gdi.setCX(headXPos[0] + gdi.scaleLength(30));
//gdi.addInput(legno, "", 2);
//gdi.setText(legno, k+1);
// gdi.disableInput(legno);
headXPos[1]=gdi.getCX();
string legType(string("LegType")+legno);
gdi.addSelection(legType, 100, 200, MultiCB);
oClass::fillLegTypes(gdi, legType);
headXPos[2]=gdi.getCX();
string startType(string("StartType")+legno);
gdi.addSelection(startType, 90, 200, MultiCB);
oClass::fillStartTypes(gdi, startType, k == 0 || !(hasRelay || multipleRaces));
headXPos[3]=gdi.getCX();
gdi.addInput(string("StartData")+legno, L"", 8, MultiCB);
if (multipleRaces) {
string multir(string("MultiR")+legno);
headXPos[4]=gdi.getCX();
gdi.addSelection(multir, 60, 200, MultiCB);
gdi.addItem(multir, legs);
}
if (hasRelay) {
headXPos[5]=gdi.getCX();
gdi.addInput(string("RestartRope")+legno, L"", 5, MultiCB);
headXPos[6]=gdi.getCX();
gdi.addInput(string("Restart")+legno, L"", 5, MultiCB);
}
gdi.dropLine(-0.1);
gdi.addButton(string("@Course")+legno, "Banor...", MultiCB);
gdi.fillDown();
gdi.popX();
gdi.dropLine(2.1);
if (k==0) { //Add headers
gdi.addString("", headYPos, headXPos[0], 0, "Str.");
gdi.addString("", headYPos, headXPos[1], 0, "Sträcktyp:");
gdi.addString("", headYPos, headXPos[2], 0, "Starttyp:");
gdi.addString("", headYPos, headXPos[3], 0, "Starttid:");
if (multipleRaces)
gdi.addString("", headYPos, headXPos[4], 0, "Löpare:");
if (hasRelay) {
gdi.addString("", headYPos, headXPos[5], 0, "Rep:");
gdi.addString("", headYPos, headXPos[6], 0, "Omstart:");
}
}
}
gdi.pushX();
gdi.fillRight();
gdi.addCheckbox("CoursePool", "Använd banpool", MultiCB, false,
"Knyt löparna till banor från en pool vid målgång.");
gdi.addCheckbox("Unordered", "Oordnade parallella sträckor", MultiCB, false,
"Tillåt löpare inom en parallell grupp att springa gruppens banor i godtycklig ordning.");
gdi.addCheckbox("LockForking", "Lås gafflingar", MultiCB, false,
"Markera för att förhindra oavsiktlig ändring av gafflingsnycklar.");
gdi.popX();
gdi.fillRight();
gdi.dropLine(1.7);
gdi.addString("FairForking", 1, "The forking is fair.");
gdi.setCX(gdi.getCX() + gdi.getLineHeight() * 5);
gdi.dropLine(-0.3);
gdi.addButton("ShowForking", "Show forking...", MultiCB);
gdi.fillDown();
gdi.addButton("DefineForking", "Define forking...", MultiCB);
gdi.popX();
RECT rc;
rc.left = cx;
rc.right = gdi.getWidth()+10;
rc.bottom = gdi.getCY()+10;
rc.top = cy;
gdi.addRectangle(rc, colorLightBlue, true, false).set3D(true);
gdi.setRestorePoint("Courses");
if (nLeg==1)
gdi.sendCtrlMessage("@Course0");
}
gdi.refresh();
}
bool TabClass::checkClassSelected(const gdioutput &gdi) const
{
if (ClassId<=0) {
gdi.alert("Ingen klass vald.");
return false;
}
else return true;
}
void TabClass::save(gdioutput &gdi, bool skipReload)
{
bool checkValid = EditChanged || gdi.isInputChanged("");
DWORD cid=ClassId;
pClass pc;
wstring name = gdi.getText("Name");
if (cid==0 && name.empty())
return;
if (name.empty())
throw std::exception("Klassen måste ha ett namn.");
bool create=false;
if (cid>0)
pc=oe->getClass(cid);
else {
pc=oe->addClass(name);
create=true;
}
if (!pc)
throw std::exception("Class not found.");
ClassId=pc->getId();
pc->setName(name);
if (gdi.hasField("StartName"))
pc->setStart(gdi.getText("StartName"));
if (gdi.hasField("ClassType"))
pc->setType(gdi.getText("ClassType"));
if (gdi.hasField("StartBlock"))
pc->setBlock(gdi.getTextNo("StartBlock"));
if (gdi.hasField("Status")) {
pc->getDI().setEnum("Status", gdi.getSelectedItem("Status").first);
}
if (gdi.hasField("CoursePool"))
pc->setCoursePool(gdi.isChecked("CoursePool"));
if (gdi.hasField("Unordered"))
pc->setUnorderedLegs(gdi.isChecked("Unordered"));
if (gdi.hasField("LockForking"))
pc->lockedForking(gdi.isChecked("LockForking"));
pc->setAllowQuickEntry(gdi.isChecked("AllowQuickEntry"));
pc->setNoTiming(gdi.isChecked("NoTiming"));
if (gdi.hasField("FreeStart"))
pc->setFreeStart(gdi.isChecked("FreeStart"));
if (gdi.hasField("IgnoreStart"))
pc->setIgnoreStartPunch(gdi.isChecked("IgnoreStart"));
if (gdi.hasField("DirectResult")) {
bool withDirect = gdi.isChecked("DirectResult");
if (withDirect && !pc->hasDirectResult() && !hasWarnedDirect) {
if (gdi.ask(L"warning:direct_result"))
hasWarnedDirect = true;
else
withDirect = false;
}
pc->setDirectResult(withDirect);
}
int crs = gdi.getSelectedItem("Courses").first;
if (crs==0) {
//Skapa ny bana...
pCourse pcourse=oe->addCourse(L"Bana " + name);
pc->setCourse(pcourse);
pc->synchronize();
return;
}
else if (crs==-2)
pc->setCourse(0);
else if (crs > 0)
pc->setCourse(oe->getCourse(crs));
if (pc->hasMultiCourse()) {
if (gdi.hasData("SimpleMulti")) {
bool sim = gdi.isChecked("CommonStart");
if (sim) {
pc->setStartType(0, STTime, true);
if (!warnDrawStartTime(gdi, gdi.getText("CommonStartTime")))
pc->setStartData(0, gdi.getText("CommonStartTime"));
}
else {
pc->setStartType(0, STDrawn, true);
}
}
else {
int nstage=pc->getNumStages();
bool needAdjust = false;
for (int k=0;k<nstage;k++) {
char legno[10];
sprintf_s(legno, "%d", k);
if (!gdi.hasField(string("LegType")+legno))
continue;
pc->setLegType(k, LegTypes(gdi.getSelectedItem(string("LegType")+legno).first));
pc->setStartType(k, StartTypes(gdi.getSelectedItem(string("StartType")+legno).first), true);
if (pc->getStartType(k) == STChange) {
int val = gdi.getSelectedItem(string("StartData")+legno).first;
if (val <= -10)
pc->setStartData(k, val + 10);
else
pc->setStartData(k, 0);
}
else {
pc->setStartData(k, gdi.getText(string("StartData")+legno));
}
string key;
key = string("Restart")+legno;
if (gdi.hasField(key))
pc->setRestartTime(k, gdi.getText(key));
key = string("RestartRope")+legno;
if (gdi.hasField(key))
pc->setRopeTime(k, gdi.getText(key));
key = string("MultiR")+legno;
if (gdi.hasField(key)) {
int mr = gdi.getSelectedItem(key).first;
if (pc->getLegRunner(k) != mr)
needAdjust = true;
pc->setLegRunner(k, mr);
}
}
if (needAdjust)
oe->adjustTeamMultiRunners(pc);
}
}
pc->addClassDefaultFee(false);
pc->updateChangedCoursePool();
pc->synchronize();
oe->reCalculateLeaderTimes(pc->getId());
set<int> cls;
cls.insert(pc->getId());
oe->reEvaluateAll(cls, true);
oe->fillClasses(gdi, "Classes", oEvent::extraDrawn, oEvent::filterNone);
EditChanged=false;
if (!skipReload) {
ClassId = 0;
selectClass(gdi, pc->getId());
}
if (checkValid) {
// Check/warn that starts blocks are set up correctly
vector<int> b;
vector<wstring> s;
oe->getStartBlocks(b, s);
oe->sanityCheck(gdi, false, pc->getId());
}
}
struct ButtonData {
ButtonData(const char *idIn,
const char *labelIn,
bool glob) : id(idIn), label(labelIn), global(glob) {}
string id;
string label;
bool global;
};
bool TabClass::loadPage(gdioutput &gdi)
{
if (!gdi.hasData("ClassPageLoaded"))
hasWarnedStartTime = false;
oe->checkDB();
oe->checkNecessaryFeatures();
gdi.selectTab(tabId);
clearPage(gdi, false);
int xp=gdi.getCX();
const int button_w=gdi.scaleLength(90);
string switchMode;
switchMode=tableMode ? "Formulärläge" : "Tabelläge";
gdi.addButton(2, 2, button_w, "SwitchMode", switchMode,
ClassesCB, "Välj vy", false, false).fixedCorner();
if (tableMode) {
Table *tbl=oe->getClassTB();
gdi.addTable(tbl, xp, 30);
return true;
}
if (showForkingGuide) {
try {
defineForking(gdi, false);
}
catch(...) {
showForkingGuide = false;
throw;
}
return true;
}
ClassConfigInfo cnf;
oe->getClassConfigurationInfo(cnf);
bool showAdvanced = oe->getPropertyInt("AdvancedClassSettings", 0) != 0;
gdi.addString("", boldLarge, "Klasser");
gdi.fillDown();
gdi.addListBox("Classes", 200, showAdvanced ? 512 : 420, ClassesCB, L"").isEdit(false).ignore(true);
gdi.setTabStops("Classes", 185);
oe->fillClasses(gdi, "Classes", oEvent::extraDrawn, oEvent::filterNone);
gdi.newColumn();
gdi.dropLine(2);
gdi.fillRight();
gdi.pushX();
gdi.addInput("Name", L"", 14, ClassesCB, L"Klassnamn:");
bool sameLineNameCourse = true;
if (showAdvanced) {
gdi.addCombo("ClassType", 100, 300, 0, L"Typ:");
oe->fillClassTypes(gdi, "ClassType");
sameLineNameCourse = false;
}
if (showMulti(false) || !sameLineNameCourse) {
gdi.dropLine(3);
gdi.popX();
}
gdi.addSelection("Courses", 120, 400, ClassesCB, L"Bana:");
oe->fillCourses(gdi, "Courses", true);
gdi.addItem("Courses", lang.tl("Ingen bana"), -2);
if (showMulti(false)) {
gdi.dropLine(0.9);
if (showMulti(true)) {
gdi.addButton("MultiCourse", "Flera banor/stafett...", ClassesCB);
}
else {
gdi.addButton("MultiCourse", "Gafflade banor...", ClassesCB);
}
gdi.disableInput("MultiCourse");
}
gdi.popX();
if (showAdvanced) {
gdi.dropLine(3);
gdi.addCombo("StartName", 120, 300, 0, L"Startnamn:");
oe->fillStarts(gdi, "StartName");
gdi.addSelection("StartBlock", 80, 300, 0, L"Startblock:");
for (int k=1;k<=100;k++) {
gdi.addItem("StartBlock", itow(k), k);
}
gdi.popX();
gdi.dropLine(3);
gdi.addSelection("Status", 200, 300, 0, L"Status:");
}
gdi.popX();
gdi.dropLine(3.5);
gdi.addCheckbox("AllowQuickEntry", "Tillåt direktanmälan", 0);
gdi.addCheckbox("NoTiming", "Utan tidtagning", 0);
if (showAdvanced) {
gdi.dropLine(2);
gdi.popX();
gdi.addCheckbox("FreeStart", "Fri starttid", 0, false, "Klassen lottas inte, startstämpling");
gdi.addCheckbox("IgnoreStart", "Ignorera startstämpling", 0, false, "Uppdatera inte starttiden vid startstämpling");
gdi.dropLine(2);
gdi.popX();
gdi.addCheckbox("DirectResult", "Resultat vid målstämpling", 0, false,
"help:DirectResult");
}
gdi.dropLine(2);
gdi.popX();
vector<ButtonData> func;
if (oe->getMeOSFeatures().hasFeature(MeOSFeatures::DrawStartList))
func.push_back(ButtonData("Draw", "Lotta / starttider...", false));
if (oe->getMeOSFeatures().hasFeature(MeOSFeatures::Bib))
func.push_back(ButtonData("Bibs", "Nummerlappar...", false));
if (oe->getMeOSFeatures().hasFeature(MeOSFeatures::DrawStartList))
func.push_back(ButtonData("DrawMode", "Lotta flera klasser", true));
if (cnf.hasTeamClass()) {
func.push_back(ButtonData("Restart", "Omstart...", true));
vector<pClass> allCls;
oe->getClasses(allCls, false);
bool unlockedClass = false;
bool lockedClass = false;
if (showAdvanced) {
for (pClass c : allCls) {
if (c->isRemoved())
continue;
if (!c->hasCoursePool() && c->hasMultiCourse()) {
if (c->lockedForking())
lockedClass = true;
else
unlockedClass = true;
}
}
if (unlockedClass) {
func.push_back(ButtonData("LockAllForks", "Lås gafflingar", true));
}
if (lockedClass) {
func.push_back(ButtonData("UnLockAllForks", "Tillåt gafflingsändringar", true));
}
}
}
if (showAdvanced) {
func.push_back(ButtonData("Merge", "Slå ihop klasser...", false));
func.push_back(ButtonData("Split", "Dela klassen...", false));
}
if (showAdvanced && oe->getMeOSFeatures().hasFeature(MeOSFeatures::Vacancy)) {
vector<pRunner> rr;
oe->getRunners(0, 0, rr, false);
bool hasVac = false;
for (size_t k = 0; k < rr.size(); k++) {
if (rr[k]->isVacant()) {
hasVac = true;
break;
}
}
if (hasVac)
func.push_back(ButtonData("RemoveVacant", "Radera vakanser", true));
}
func.push_back(ButtonData("QuickSettings", "Snabbinställningar", true));
RECT funRect;
funRect.right = gdi.getCX() - 7;
funRect.top = gdi.getCY() - 2;
funRect.left = 0;
gdi.dropLine(0.5);
gdi.fillDown();
gdi.addString("", fontMediumPlus, "Funktioner");
gdi.dropLine(0.3);
gdi.pushX();
gdi.fillRight();
int xlimit = gdi.getWidth() - button_w/2;
for (size_t k = 0; k < func.size(); k++) {
ButtonInfo &bi = gdi.addButton(func[k].id, func[k].label, ClassesCB);
if (!func[k].global)
bi.isEdit(true);
funRect.left = max<int>(funRect.left, gdi.getCX() + 7);
if ( gdi.getCX() > xlimit && k+1 < func.size()) {
gdi.popX();
gdi.dropLine(2.5);
}
}
gdi.dropLine(2.5);
funRect.bottom = gdi.getCY();
gdi.addRectangle(funRect, colorLightGreen);
gdi.popX();
gdi.dropLine(0.5);
gdi.fillRight();
gdi.addButton("Save", "Spara", ClassesCB).setDefault();
gdi.disableInput("Save");
gdi.addButton("Remove", "Radera", ClassesCB);
gdi.disableInput("Remove");
gdi.addButton("Add", "Ny klass", ClassesCB);
gdi.popX();
gdi.fillDown();
gdi.dropLine(3);
gdi.addCheckbox("UseAdvanced", "Visa avancerade funktioner", ClassesCB, showAdvanced).isEdit(false);
gdi.setOnClearCb(ClassesCB);
gdi.setRestorePoint();
gdi.setCX(xp);
gdi.setCY(gdi.getHeight());
gdi.addString("", 10, "help:26963");
selectClass(gdi, ClassId);
EditChanged=false;
gdi.refresh();
return true;
}
bool TabClass::showMulti(bool singleOnly) const {
const MeOSFeatures &mf = oe->getMeOSFeatures();
if (!singleOnly)
return mf.hasFeature(MeOSFeatures::Relay) || mf.hasFeature(MeOSFeatures::Patrol) || mf.hasFeature(MeOSFeatures::ForkedIndividual);
else
return mf.hasFeature(MeOSFeatures::Relay) || mf.hasFeature(MeOSFeatures::Patrol) || mf.hasFeature(MeOSFeatures::MultipleRaces);
}
static int classSettingsCB(gdioutput *gdi, int type, void *data)
{
TabClass &tc = dynamic_cast<TabClass &>(*gdi->getTabs().get(TClassTab));
static wstring lastStart = L"Start 1";
if (type==GUI_INPUT) {
InputInfo ii=*(InputInfo *)data;
if (ii.id.substr(0,4) == "Strt") {
lastStart = ii.text;
}
}
else if (type == GUI_FOCUS) {
InputInfo ii=*(InputInfo *)data;
if (ii.id.substr(0,4) == "Strt") {
if (ii.text.empty()) {
gdi->setText(ii.id, lastStart);
gdi->setInputFocus(ii.id, true);
}
}
}
else if (type == GUI_BUTTON) {
ButtonInfo bi = *(ButtonInfo*)data;
if (bi.id == "SaveCS") {
tc.saveClassSettingsTable(*gdi);
}
}
else if (type == GUI_CLEAR) {
tc.saveClassSettingsTable(*gdi);
return 1;
}
return 0;
}
void TabClass::saveClassSettingsTable(gdioutput &gdi) {
set<int> modifiedFee;
bool modifiedBib = false;
saveClassSettingsTable(gdi, modifiedFee, modifiedBib);
oe->synchronize(true);
if (gdi.hasField("BibGap")) {
int gap = gdi.getTextNo("BibGap");
if (oe->getBibClassGap() != gap) {
oe->setBibClassGap(gap);
modifiedBib = true;
}
}
if (!modifiedFee.empty() && oe->getNumRunners() > 0) {
bool updateFee = gdi.ask(L"ask:changedclassfee");
if (updateFee)
oe->applyEventFees(false, true, false, modifiedFee);
}
if (modifiedBib && gdi.ask(L"Vill du uppdatera alla nummerlappar?")) {
oe->addAutoBib();
}
oe->synchronize(true);
gdi.sendCtrlMessage("Cancel");
}
void TabClass::prepareForDrawing(gdioutput &gdi) {
clearPage(gdi, false);
gdi.addString("", 2, "Klassinställningar");
int baseLine = gdi.getCY();
gdi.addString("", 10, "help:59395");
gdi.pushX();
int by = gdi.getCY();
gdi.setCX(gdi.getWidth());
gdi.setCY(baseLine);
gdi.addString("", 10, "help:59395_more");
gdi.setCY(max(gdi.getCY(), by));
gdi.popX();
gdi.dropLine();
if (oe->getMeOSFeatures().hasFeature(MeOSFeatures::Bib)) {
gdi.fillRight();
gdi.addString("", 0, "Antal reserverade nummerlappsnummer mellan klasser:");
gdi.dropLine(-0.1);
gdi.addInput("BibGap", itow(oe->getBibClassGap()), 5);
gdi.popX();
gdi.dropLine(1.5);
gdi.fillDown();
}
getClassSettingsTable(gdi, classSettingsCB);
gdi.dropLine();
gdi.fillRight();
gdi.addButton("SaveCS", "Spara", classSettingsCB);
gdi.addButton("Cancel", "Avbryt", ClassesCB);
gdi.refresh();
}
void TabClass::drawDialog(gdioutput &gdi, DrawMethod method, const oClass &pc) {
oe->setProperty("DefaultDrawMethod", method);
if (lastDrawMethod == method)
return;
if (lastDrawMethod == DMPursuit && method == DMReversePursuit)
return;
if (lastDrawMethod == DMReversePursuit && method == DMPursuit)
return;
if (lastDrawMethod == DMRandom && method == DMSOFT)
return;
if (lastDrawMethod == DMSOFT && method == DMRandom)
return;
int firstStart = 3600,
interval = 120,
vac = _wtoi(lastNumVac.c_str());
int pairSize = lastPairSize;
if (gdi.hasField("FirstStart"))
firstStart = oe->getRelativeTime(gdi.getText("FirstStart"));
else if (!lastFirstStart.empty())
firstStart = oe->getRelativeTime(lastFirstStart);
if (gdi.hasField("Interval"))
interval = convertAbsoluteTimeMS(gdi.getText("Interval"));
else if (!lastInterval.empty())
interval = convertAbsoluteTimeMS(lastInterval);
if (gdi.hasField("PairSize")) {
pairSize = gdi.getSelectedItem("PairSize").first;
}
gdi.restore("MultiDayDraw", false);
const bool multiDay = oe->hasPrevStage();
if (method == DMSeeded) {
gdi.addString("", 10, "help:seeding_info");
gdi.dropLine(1);
gdi.pushX();
gdi.fillRight();
ListBoxInfo &seedmethod = gdi.addSelection("SeedMethod", 120, 100, 0, L"Seedningskälla:");
vector< pair<wstring, size_t> > methods;
oClass::getSeedingMethods(methods);
gdi.addItem("SeedMethod", methods);
if (lastSeedMethod == -1)
gdi.selectFirstItem("SeedMethod");
else
gdi.selectItemByData("SeedMethod", lastSeedMethod);
seedmethod.setSynchData(&lastSeedMethod);
gdi.addInput("SeedGroups", lastSeedGroups, 32, 0, L"Seedningsgrupper:",
L"Ange en gruppstorlek (som repeteras) eller flera kommaseparerade gruppstorlekar").
setSynchData(&lastSeedGroups);
gdi.fillDown();
gdi.popX();
gdi.dropLine(3);
gdi.addCheckbox("PreventClubNb", "Hindra att deltagare från samma klubb startar på angränsande tider",
0, lastSeedPreventClubNb).setSynchData(&lastSeedPreventClubNb);
gdi.addCheckbox("ReverseSeedning", "Låt de bästa start först", 0, lastSeedReverse).
setSynchData(&lastSeedReverse);
}
else {
gdi.popX();
gdi.addString("", 10, "help:41641");
gdi.dropLine(1);
}
if (method == DMRandom || method == DMSOFT || method == DMPursuit
|| method == DMReversePursuit || method == DMSeeded) {
gdi.addSelection("PairSize", 150, 200, 0, L"Tillämpa parstart:").setSynchData(&lastPairSize);
gdi.addItem("PairSize", getPairOptions());
gdi.selectItemByData("PairSize", pairSize);
}
gdi.fillRight();
gdi.addInput("FirstStart", oe->getAbsTime(firstStart), 10, 0, L"Första start:").setSynchData(&lastFirstStart);
if (method == DMPursuit || method == DMReversePursuit) {
gdi.addInput("MaxAfter", lastMaxAfter, 10, 0, L"Maxtid efter:", L"Maximal tid efter ledaren för att delta i jaktstart").setSynchData(&lastMaxAfter);
gdi.addInput("TimeRestart", oe->getAbsTime(firstStart + 3600), 8, 0, L"Första omstartstid:");
gdi.addInput("ScaleFactor", lastScaleFactor, 8, 0, L"Tidsskalning:").setSynchData(&lastScaleFactor);
}
if (method != DMSimultaneous)
gdi.addInput("Interval", formatTime(interval), 10, 0, L"Startintervall (min):").setSynchData(&lastInterval);
if (method == DMRandom || method == DMSOFT || method == DMClumped)
gdi.addInput("Vacanses", itow(vac), 10, 0, L"Antal vakanser:").setSynchData(&lastNumVac);
if ((method == DMRandom || method == DMSOFT || method == DMSeeded) && pc.getNumStages() > 1 && pc.getClassType() != oClassPatrol) {
gdi.addSelection("Leg", 90, 100, 0, L"Sträcka:", L"Sträcka att lotta");
for (unsigned k = 0; k<pc.getNumStages(); k++)
gdi.addItem("Leg", lang.tl("Sträcka X#" + itos(k+1)), k);
gdi.selectFirstItem("Leg");
}
if (int(method) < 10) {
gdi.popX();
gdi.dropLine(3.5);
gdi.fillRight();
gdi.addCheckbox("HandleBibs", "Tilldela nummerlappar:", ClassesCB, lastHandleBibs).setSynchData(&lastHandleBibs);
gdi.dropLine(-0.2);
gdi.addInput("Bib", L"", 10, 0, L"", L"Mata in första nummerlappsnummer, eller blankt för att ta bort nummerlappar");
gdi.disableInput("Bib");
gdi.fillDown();
gdi.dropLine(2.5);
gdi.popX();
}
else {
gdi.popX();
gdi.dropLine(3.5);
}
gdi.fillRight();
if (method != DMSimultaneous)
gdi.addButton("DoDraw", "Lotta klassen", ClassesCB, "Lotta om hela klassen");
else
gdi.addButton("DoDraw", "Tilldela", ClassesCB, "Tilldela starttider");
if (method == DMRandom || method == DMSOFT) {
gdi.addButton("DoDrawBefore", "Ej lottade, före", ClassesCB, "Lotta löpare som saknar starttid");
gdi.addButton("DoDrawAfter", "Ej lottade, efter", ClassesCB, "Lotta löpare som saknar starttid");
}
gdi.addButton("DoDeleteStart", "Radera starttider", ClassesCB);
gdi.fillDown();
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.popX();
gdi.dropLine();
setMultiDayClass(gdi, multiDay, method);
EditChanged=false;
gdi.refresh();
lastDrawMethod = method;
}
void TabClass::setMultiDayClass(gdioutput &gdi, bool hasMulti, DrawMethod defaultMethod) {
gdi.clearList("Method");
gdi.addItem("Method", lang.tl("Lottning"), DMRandom);
gdi.addItem("Method", lang.tl("SOFT-lottning"), DMSOFT);
gdi.addItem("Method", lang.tl("Klungstart"), DMClumped);
gdi.addItem("Method", lang.tl("Gemensam start"), DMSimultaneous);
gdi.addItem("Method", lang.tl("Seedad lottning"), DMSeeded);
if (hasMulti) {
gdi.addItem("Method", lang.tl("Jaktstart"), DMPursuit);
gdi.addItem("Method", lang.tl("Omvänd jaktstart"), DMReversePursuit);
}
else if (defaultMethod > 10)
defaultMethod = getDefaultMethod(hasMulti);
gdi.selectItemByData("Method", defaultMethod);
if (gdi.hasField("Vacanses")) {
gdi.setInputStatus("Vacanses", !hasMulti);
gdi.setInputStatus("HandleBibs", !hasMulti);
if (hasMulti) {
gdi.check("HandleBibs", false);
gdi.setInputStatus("Bib", false);
}
}
if (gdi.hasField("DoDrawBefore")) {
gdi.setInputStatus("DoDrawBefore", !hasMulti);
gdi.setInputStatus("DoDrawAfter", !hasMulti);
}
}
void TabClass::pursuitDialog(gdioutput &gdi) {
clearPage(gdi, false);
gdi.addString("", boldLarge, "Jaktstart");
gdi.dropLine();
vector<pClass> cls;
oe->getClasses(cls, true);
gdi.setRestorePoint("Pursuit");
gdi.pushX();
gdi.fillRight();
gdi.addInput("MaxAfter", formatTime(pSavedDepth), 10, 0, L"Maxtid efter:", L"Maximal tid efter ledaren för att delta i jaktstart");
gdi.addInput("TimeRestart", L"+" + formatTime(pFirstRestart), 8, 0, L"Första omstartstid:", L"Ange tiden relativt klassens första start");
gdi.addInput("Interval", formatTime(pInterval), 8, 0, L"Startintervall:", L"Ange startintervall för minutstart");
wchar_t bf[32];
swprintf_s(bf, L"%f", pTimeScaling);
gdi.addInput("ScaleFactor", bf, 8, 0, L"Tidsskalning:");
gdi.dropLine(4);
gdi.popX();
gdi.fillDown();
//xxx
//gdi.addCheckbox("Pairwise", "Tillämpa parstart", 0, false);
gdi.addSelection("PairSize", 150, 200, 0, L"Tillämpa parstart:");
gdi.addItem("PairSize", getPairOptions());
gdi.selectItemByData("PairSize", 1);
int cx = gdi.getCX();
int cy = gdi.getCY();
const int len5 = gdi.scaleLength(5);
const int len40 = gdi.scaleLength(30);
const int len200 = gdi.scaleLength(200);
gdi.addString("", cy, cx, 1, "Välj klasser");
gdi.addString("", cy, cx + len200 + len40, 1, "Första starttid");
cy += gdi.getLineHeight()*2;
for (size_t k = 0; k<cls.size(); k++) {
map<int, PursuitSettings>::iterator st = pSettings.find(cls[k]->getId());
if (st == pSettings.end()) {
pSettings.insert(make_pair(cls[k]->getId(), PursuitSettings(*cls[k])));
st = pSettings.find(cls[k]->getId());
}
PursuitSettings &ps = st->second;
int fs = cls[k]->getDrawFirstStart();
if (fs > 0)
ps.firstTime = fs;
ButtonInfo &bi = gdi.addCheckbox(cx, cy + len5, "PLUse" + itos(k), "", ClassesCB, ps.use);
bi.setExtra(cls[k]->getId());
gdi.addStringUT(cy, cx + len40, 0, cls[k]->getName(), len200);
gdi.addInput(cx + len200 + len40, cy, "First" + itos(k), oe->getAbsTime(ps.firstTime), 8);
if (!ps.use)
gdi.disableInput(("First" + itos(k)).c_str());
cy += int(gdi.getLineHeight()*1.8);
}
gdi.dropLine();
gdi.fillRight();
gdi.addButton("SelectAllNoneP", "Välj alla", ClassesCB).setExtra(1);
gdi.addButton("SelectAllNoneP", "Välj ingen", ClassesCB).setExtra(0);
gdi.popX();
gdi.dropLine(3);
RECT rc;
rc.left = gdi.getCX();
rc.top = gdi.getCY();
rc.bottom = rc.top + gdi.getButtonHeight() + gdi.scaleLength(17);
gdi.setCX(rc.left + gdi.scaleLength(10));
gdi.setCY(rc.top + gdi.scaleLength(10));
gdi.addButton("DoPursuit", "Jaktstart", ClassesCB).setDefault().setExtra(1);
gdi.addButton("DoPursuit", "Omvänd jaktstart", ClassesCB).setExtra(2);
rc.right = gdi.getCX() + gdi.scaleLength(5);
gdi.addRectangle(rc, colorLightGreen);
gdi.setCX(rc.right + gdi.scaleLength(10));
gdi.addButton("SavePursuit", "Spara starttider", ClassesCB, "Spara inmatade tider i tävlingen utan att tilldela starttider.");
gdi.addButton("CancelPursuit", "Återgå", ClassesCB).setCancel();
gdi.refresh();
}
void TabClass::showClassSelection(gdioutput &gdi, int &bx, int &by, GUICALLBACK classesCB) const {
gdi.pushY();
int cx = gdi.getCX();
int width = gdi.scaleLength(230);
gdi.addListBox("Classes", 200, 400, classesCB, L"Klasser:", L"", true);
gdi.setTabStops("Classes", 185);
gdi.fillRight();
gdi.pushX();
gdi.addButton("SelectAll", "Välj allt", ClassesCB,
"Välj alla klasser").isEdit(true);
gdi.addButton("SelectMisses", "Saknad starttid", ClassesCB,
"Välj klasser där någon löpare saknar starttid").isEdit(true);
gdi.dropLine(2.3);
gdi.popX();
gdi.addButton("SelectUndrawn", "Ej lottade", ClassesCB,
"Välj klasser där alla löpare saknar starttid").isEdit(true);
gdi.fillDown();
gdi.addButton("SelectNone", "Välj inget", ClassesCB,
"Avmarkera allt").isEdit(true);
gdi.popX();
vector<int> blocks;
vector<wstring> starts;
oe->getStartBlocks(blocks, starts);
map<wstring, int> sstart;
for (size_t k = 0; k < starts.size(); k++) {
sstart.insert(make_pair(starts[k], k));
}
if (sstart.size() > 1) {
gdi.fillRight();
int cnt = 0;
for (map<wstring, int>::reverse_iterator it = sstart.rbegin(); it != sstart.rend(); ++it) {
if ((cnt & 1)==0 && cnt>0) {
gdi.dropLine(2);
gdi.popX();
}
wstring name = it->first;
if (name.empty())
name = lang.tl(L"övriga");
gdi.addButton("SelectStart", L"Välj X#" + name, ClassesCB, L"").isEdit(true).setExtra(it->second);
cnt++;
}
gdi.dropLine(2.5);
gdi.popX();
gdi.fillDown();
}
oe->fillClasses(gdi, "Classes", oEvent::extraDrawn, oEvent::filterNone);
by = gdi.getCY()+gdi.getLineHeight();
bx = gdi.getCX();
//gdi.newColumn();
gdi.setCX(cx+width);
gdi.popY();
}
void TabClass::enableLoadSettings(gdioutput &gdi) {
if (!gdi.hasField("LoadSettings"))
return;
set<int> sel;
gdi.getSelection("Classes", sel);
bool ok = !sel.empty();
gdi.setInputStatus("PrepareDrawAll", ok);
gdi.setInputStatus("EraseStartAll", ok);
ok = false;
for (set<int>::iterator it = sel.begin(); it != sel.end(); ++it) {
pClass pc = oe->getClass(*it);
if (pc) {
if (pc->getDrawFirstStart() > 0 && pc->getDrawInterval() > 0) {
ok = true;
break;
}
}
}
gdi.setInputStatus("LoadSettings", ok);
}
void TabClass::simultaneous(int classId, const wstring &time) {
pClass pc = oe->getClass(classId);
if (!pc)
throw exception();
pc->getNumStages();
if (pc->getNumStages() == 0) {
pCourse crs = pc->getCourse();
pc->setNumStages(1);
if (crs)
pc->addStageCourse(0, crs->getId());
}
pc->setStartType(0, STTime, false);
pc->setStartData(0, time);
pc->synchronize(true);
pc->forceShowMultiDialog(false);
oe->reCalculateLeaderTimes(pc->getId());
set<int> cls;
cls.insert(pc->getId());
oe->reEvaluateAll(cls, true);
}
const wchar_t *TabClass::getCourseLabel(bool pool) {
if (pool)
return L"Banpool:";
else
return L"Sträckans banor:";
}
void TabClass::selectCourses(gdioutput &gdi, int legNo) {
gdi.restore("Courses", false);
gdi.setRestorePoint("Courses");
wchar_t bf[128];
pClass pc=oe->getClass(ClassId);
if (!pc) {
gdi.refresh();
return;
}
currentStage = legNo;
gdi.dropLine();
gdi.pushX();
gdi.fillRight();
bool simpleView = pc->getNumStages() == 1;
if (!simpleView) {
swprintf_s(bf, lang.tl("Banor för %s, sträcka %d").c_str(), pc->getName().c_str(), legNo+1);
gdi.addStringUT(1, bf);
ButtonInfo &bi1 = gdi.addButton("@Course" + itos(legNo-1), "<< Föregående", MultiCB);
if (legNo<=0)
gdi.disableInput(bi1.id.c_str());
ButtonInfo &bi2 = gdi.addButton("@Course" + itos(legNo+1), "Nästa >>", MultiCB);
if (unsigned(legNo + 1) >= pc->getNumStages())
gdi.disableInput(bi2.id.c_str());
gdi.popX();
gdi.dropLine(2.5);
}
gdi.fillRight();
int x1=gdi.getCX();
gdi.addListBox("StageCourses", 180, 200, MultiCB, getCourseLabel(pc->hasCoursePool())).ignore(true);
pc->fillStageCourses(gdi, currentStage, "StageCourses");
int x2=gdi.getCX();
gdi.fillDown();
gdi.addListBox("MCourses", 180, 200, MultiCB, L"Banor:").ignore(true);
oe->fillCourses(gdi, "MCourses", true);
gdi.setCX(x1);
gdi.fillRight();
gdi.addButton("MRemove", "Ta bort markerad >>", MultiCB);
gdi.setCX(x2);
gdi.fillDown();
gdi.addButton("MAdd", "<< Lägg till", MultiCB);
gdi.setCX(x1);
gdi.refresh();
if (pc->getNumStages() > 1)
gdi.scrollTo(gdi.getCX(), gdi.getCY());
}
void TabClass::updateFairForking(gdioutput &gdi, pClass pc) const {
vector< vector<int> > forks;
vector< vector<int> > forksC;
set< pair<int, int> > unfairLegs;
if (pc->checkForking(forksC, forks, unfairLegs)) {
BaseInfo *bi = gdi.setText("FairForking", gdi.getText("FairForking"), false);
TextInfo &text = dynamic_cast<TextInfo &>(*bi);
text.setColor(colorGreen);
gdi.setText("FairForking", lang.tl("The forking is fair."), true);
}
else {
BaseInfo *bi = gdi.setText("FairForking", gdi.getText("FairForking"), false);
TextInfo &text = dynamic_cast<TextInfo &>(*bi);
text.setColor(colorRed);
gdi.setText("FairForking", lang.tl("The forking is not fair."), true);
}
}
void TabClass::defineForking(gdioutput &gdi, bool clearSettings) {
pClass pc = oe->getClass(ClassId);
if (clearSettings) {
forkingSetup.clear();
forkingSetup.resize(pc->getNumStages());
}
else if (forkingSetup.size() != pc->getNumStages())
throw meosException("Internal error");
showForkingGuide = true;
gdi.clearPage(false);
int tx = gdi.getCX();
int ty = gdi.getCY();
gdi.dropLine(2);
gdi.pushY();
gdi.addListBox("AllCourses", 180, 300, 0, L"Banor:", L"", true);
oe->fillCourses(gdi, "AllCourses", true);
int bxp = gdi.getCX();
int byp = gdi.getCY();
gdi.fillDown();
gdi.newColumn();
gdi.popY();
gdi.addListBox("AllStages", 180, 300, MultiCB, L"Legs:", L"", true);
int ns = pc->getNumStages();
gdi.newColumn();
gdi.fillDown();
gdi.popY();
gdi.addButton("AssignCourses", "Assign selected courses to selected legs", MultiCB);
gdi.disableInput("AssignCourses");
gdi.dropLine();
gdi.addString("", boldText, "Forking setup");
gdi.dropLine(0.5);
for (int k = 0; k < ns; k++) {
LegTypes lt = pc->getLegType(k);
if (lt != LTIgnore) {
gdi.addString("leg"+ itos(k), 0, "Leg X: Do not modify.#" + itos(k+1));
gdi.addItem("AllStages", lang.tl("Leg X#" + itos(k+1)), k);
}
}
gdi.dropLine();
gdi.fillRight();
gdi.addButton("ApplyForking", "Calculate and apply forking", MultiCB);
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.disableInput("ApplyForking");
gdi.setCX(bxp);
gdi.setCY(byp);
gdi.fillDown();
gdi.addButton("ClearCourses", "Clear selections", MultiCB);
gdi.addString("", 10, "help:assignforking");
gdi.addString("", ty, tx, boldLarge, L"Assign courses and apply forking to X#" + pc->getName());
if (!clearSettings)
gdi.sendCtrlMessage("AssignCourses");
gdi.refresh();
}
void TabClass::getClassSettingsTable(gdioutput &gdi, GUICALLBACK cb) {
vector<pClass> cls;
oe->getClasses(cls, true);
int yp = gdi.getCY();
int a = gdi.scaleLength(160);
int b = gdi.scaleLength(250);
int c = gdi.scaleLength(300);
int d = gdi.scaleLength(350);
int e = gdi.scaleLength(510);
int et = gdi.scaleLength(605);
int f = gdi.scaleLength(510);
int g = gdi.scaleLength(535);
int ek1 = 0, ekextra = 0;
bool useEco = oe->getMeOSFeatures().hasFeature(MeOSFeatures::Economy);
gdi.setOnClearCb(cb);
if (useEco) {
ek1 = gdi.scaleLength(70);
ekextra = gdi.scaleLength(35);
d += 4 * ek1 + ekextra;
e += 4 * ek1 + ekextra;
et += 4 * ek1 + ekextra;
f += 4 * ek1 + ekextra;
g += 4 * ek1 + ekextra;
gdi.addString("", yp, c+ek1, 1, "Avgift");
gdi.addString("", yp, c+2*ek1, 1, "Sen avgift");
gdi.addString("", yp, c+3*ek1, 1, "Red. avgift");
gdi.addString("", yp, c+4*ek1, 1, "Sen red. avgift");
}
gdi.addString("", yp, gdi.getCX(), 1, "Klass");
gdi.addString("", yp, a, 1, "Start");
gdi.addString("", yp, b, 1, "Block");
gdi.addString("", yp, c, 1, "Index");
gdi.addString("", yp, d, 1, "Bana");
const bool useBibs = oe->getMeOSFeatures().hasFeature(MeOSFeatures::Bib);
const bool useTeam = oe->hasTeam();
vector< pair<wstring, size_t> > bibOptions;
vector< pair<wstring, size_t> > bibTeamOptions;
if (useBibs) {
gdi.addString("", yp, e, 1, "Nummerlapp");
bibOptions.push_back(make_pair(lang.tl("Manuell"), 0));
bibOptions.push_back(make_pair(lang.tl("Löpande"), 1));
bibOptions.push_back(make_pair(lang.tl("Ingen"), 2));
int bibW = gdi.scaleLength(100);
if (useTeam) {
gdi.addString("", yp, et, 1, "Lagmedlem");
bibTeamOptions.push_back(make_pair(lang.tl("Oberoende"), BibFree));
bibTeamOptions.push_back(make_pair(lang.tl("Samma"), BibSame));
bibTeamOptions.push_back(make_pair(lang.tl("Ökande"), BibAdd));
bibTeamOptions.push_back(make_pair(lang.tl("Sträcka"), BibLeg));
bibW += gdi.scaleLength(85);
}
f += bibW;
g += bibW;
}
gdi.addString("", yp, f, 1, "Direktanmälan");
vector< pair<wstring,size_t> > arg;
oe->fillCourses(arg, true);
for (size_t k = 0; k < cls.size(); k++) {
pClass it = cls[k];
int cyp = gdi.getCY();
string id = itos(it->getId());
gdi.addStringUT(0, it->getName(), 0);
gdi.addInput(a, cyp, "Strt"+id, it->getStart(), 7, cb);
wstring blk = it->getBlock()>0 ? itow(it->getBlock()) : L"";
gdi.addInput(b, cyp, "Blck"+id, blk, 4);
gdi.addInput(c, cyp, "Sort"+id, itow(it->getDCI().getInt("SortIndex")), 4);
if (useEco) {
gdi.addInput(c + ek1, cyp, "Fee"+id, oe->formatCurrency(it->getDCI().getInt("ClassFee")), 5);
gdi.addInput(c + 2*ek1, cyp, "LateFee"+id, oe->formatCurrency(it->getDCI().getInt("HighClassFee")), 5);
gdi.addInput(c + 3*ek1, cyp, "RedFee"+id, oe->formatCurrency(it->getDCI().getInt("ClassFeeRed")), 5);
gdi.addInput(c + 4*ek1, cyp, "RedLateFee"+id, oe->formatCurrency(it->getDCI().getInt("HighClassFeeRed")), 5);
}
string crs = "Cors"+id;
gdi.addSelection(d, cyp, crs, 150, 400);
if (it->hasTrueMultiCourse()) {
gdi.addItem(crs, lang.tl("Flera banor"), -5);
gdi.selectItemByData(crs.c_str(), -5);
gdi.disableInput(crs.c_str());
}
else {
gdi.addItem(crs, arg);
gdi.selectItemByData(crs.c_str(), it->getCourseId());
}
if (useBibs) {
gdi.addCombo(e, cyp, "Bib" + id, 90, 100, 0, L"", L"Ange löpande numrering eller första nummer i klassen.");
gdi.addItem("Bib" + id, bibOptions);
wstring bib = it->getDCI().getString("Bib");
AutoBibType bt = it->getAutoBibType();
if (bt != AutoBibExplicit)
gdi.selectItemByData("Bib"+ id, bt);
else
gdi.setText("Bib"+ id, bib);
if (useTeam && it->getNumDistinctRunners() > 1) {
gdi.addSelection(et, cyp, "BibTeam" + id, 80, 100, 0, L"", L"Ange relation mellan lagets och deltagarnas nummerlappar.");
gdi.addItem("BibTeam" + id, bibTeamOptions);
gdi.selectItemByData("BibTeam" + id, it->getBibMode());
}
}
gdi.addCheckbox(g, cyp, "Dirc"+id, " ", 0, it->getAllowQuickEntry());
gdi.dropLine(-0.3);
}
}
void TabClass::saveClassSettingsTable(gdioutput &gdi, set<int> &classModifiedFee, bool &modifiedBib) {
vector<pClass> cls;
oe->getClasses(cls, true);
classModifiedFee.clear();
modifiedBib = false;
for (size_t k = 0; k < cls.size(); k++) {
pClass it = cls[k];
string id = itos(it->getId());
wstring start = gdi.getText("Strt"+id);
int block = gdi.getTextNo("Blck"+id);
int sort = gdi.getTextNo("Sort"+id);
if (gdi.hasField("Fee" + id)) {
int fee = oe->interpretCurrency(gdi.getText("Fee"+id));
int latefee = oe->interpretCurrency(gdi.getText("LateFee"+id));
int feered = oe->interpretCurrency(gdi.getText("RedFee"+id));
int latefeered = oe->interpretCurrency(gdi.getText("RedLateFee"+id));
int oFee = it->getDCI().getInt("ClassFee");
int oLateFee = it->getDCI().getInt("HighClassFee");
int oFeeRed = it->getDCI().getInt("ClassFeeRed");
int oLateFeeRed = it->getDCI().getInt("HighClassFeeRed");
if (oFee != fee || oLateFee != latefee ||
oFeeRed != feered || oLateFeeRed != latefeered)
classModifiedFee.insert(it->getId());
it->getDI().setInt("ClassFee", fee);
it->getDI().setInt("HighClassFee", latefee);
it->getDI().setInt("ClassFeeRed", feered);
it->getDI().setInt("HighClassFeeRed", latefeered);
}
if (gdi.hasField("Bib" + id)) {
ListBoxInfo lbi;
bool mod = false;
if (gdi.getSelectedItem("Bib" + id, lbi)) {
mod = it->getDI().setString("Bib", getBibCode(AutoBibType(lbi.data), L"1"));
}
else {
const wstring &v = gdi.getText("Bib" + id);
mod = it->getDI().setString("Bib", v);
}
modifiedBib |= mod;
if (gdi.hasField("BibTeam" + id)) {
ListBoxInfo lbi_bib;
if (gdi.getSelectedItem("BibTeam" + id, lbi_bib)) {
if (it->getBibMode() != lbi_bib.data)
modifiedBib = true;
it->setBibMode(BibMode(lbi_bib.data));
}
}
}
int courseId = 0;
ListBoxInfo lbi;
if (gdi.getSelectedItem("Cors"+id, lbi))
courseId = lbi.data;
bool direct = gdi.isChecked("Dirc"+id);
it->setStart(start);
it->setBlock(block);
if (courseId != -5)
it->setCourse(oe->getCourse(courseId));
it->getDI().setInt("SortIndex", sort);
it->setAllowQuickEntry(direct);
}
}
wstring TabClass::getBibCode(AutoBibType bt, const wstring &key) {
if (bt == AutoBibManual)
return L"";
else if (bt == AutoBibConsecutive)
return L"*";
else if (bt == AutoBibNone)
return L"-";
else
return key;
}
void TabClass::updateStartData(gdioutput &gdi, pClass pc, int leg, bool updateDependent, bool forceWrite) {
string sdKey = "StartData"+itos(leg);
BaseInfo &sdataBase = gdi.getBaseInfo(sdKey.c_str());
StartTypes st = pc->getStartType(leg);
if (st == STChange) {
if (typeid(sdataBase) != typeid(ListBoxInfo)) {
InputInfo sdII = dynamic_cast<InputInfo &>(sdataBase);
gdi.removeControl(sdKey);
gdi.addSelection(sdII.getX(), sdII.getY(), sdKey, sdII.getWidth(), 200, MultiCB);
setParallelOptions(sdKey, gdi, pc, leg);
}
else if (forceWrite) {
setParallelOptions(sdKey, gdi, pc, leg);
}
}
else {
if (typeid(sdataBase) != typeid(InputInfo)) {
ListBoxInfo sdLBI = dynamic_cast<ListBoxInfo &>(sdataBase);
gdi.removeControl(sdKey);
string val = "-";
gdi.addInput(sdLBI.getX(), sdLBI.getY(), sdKey, pc->getStartDataS(leg), 8, MultiCB);
}
else if (forceWrite) {
gdi.setText(sdKey, pc->getStartDataS(leg), true);
}
gdi.setInputStatus(sdKey, !pc->startdataIgnored(leg));
}
if (updateDependent) {
for (size_t j = 0; j < pc->getNumStages(); j++) {
if (j != leg && pc->getStartType(leg) == STChange) {
setParallelOptions("StartData"+itos(j), gdi, pc, j);
}
}
}
}
void TabClass::setParallelOptions(const string &sdKey, gdioutput &gdi, pClass pc, int legno) {
int baseLeg = legno;
while (baseLeg > 0 && pc->isParallel(baseLeg))
baseLeg--;
baseLeg--;
int sd = pc->getStartData(legno);
vector< pair<wstring, size_t> > opt;
int defKey = 0;
opt.push_back(make_pair(lang.tl("Ordnat"), 0));
for (int k = 0; k <= baseLeg; k++) {
if (!pc->isOptional(k)) {
opt.push_back(make_pair(lang.tl("Str. X#" + itos(k+1)), (k-legno) - 10));
if (sd == k-legno)
defKey = sd - 10;
}
}
if (defKey == 0) {
pc->setStartData(legno, 0);
}
gdi.addItem(sdKey, opt);
gdi.selectItemByData(sdKey, defKey);
}
void TabClass::updateSplitDistribution(gdioutput &gdi, int num, int tot) const {
gdi.restore("SplitDistr", false);
gdi.setRestorePoint("SplitDistr");
vector<int> distr;
for (int k = 0; k < num; k++) {
int frac = tot / (num-k);
distr.push_back(frac);
tot -= frac;
}
sort(distr.rbegin(), distr.rend());
gdi.dropLine();
gdi.fillDown();
gdi.pushX();
for (size_t k = 0; k < distr.size(); k++) {
int yp = gdi.getCY();
int xp = gdi.getCX();
gdi.addString("", yp, xp, 0, "Klass X:#" + itos(k+1));
gdi.addInput(xp + gdi.scaleLength(100), yp, "CLS" + itos(k), itow(distr[k]), 4);
gdi.popX();
}
gdi.dropLine(1.5);
gdi.fillRight();
gdi.popX();
gdi.addButton("DoSplit", "Dela", ClassesCB).setDefault();
gdi.addButton("Cancel", "Avbryt", ClassesCB).setCancel();
gdi.popX();
gdi.refresh();
}
DrawMethod TabClass::getDefaultMethod(bool allowPursuit) const {
int dm = oe->getPropertyInt("DefaultDrawMethod", DMSOFT);
if (!allowPursuit) {
if (dm == DMRandom)
return DMRandom;
else
return DMSOFT;
}
else {
switch (dm) {
case DMRandom:
return DMRandom;
case DMPursuit:
return DMPursuit;
case DMReversePursuit:
return DMReversePursuit;
case DMSeeded:
return DMSeeded;
default:
return DMSOFT;
}
}
}
vector< pair<wstring, size_t> > TabClass::getPairOptions() {
vector< pair<wstring, size_t> > res;
res.push_back(make_pair(lang.tl("Ingen parstart"), 1));
res.push_back(make_pair(lang.tl("Parvis (två och två)"), 2));
for (int j = 3; j <= 10; j++) {
res.push_back(make_pair(lang.tl("X och Y[N by N]#" + itos(j) + "#" + itos(j)), j));
}
return res;
}
void TabClass::readDrawInfo(gdioutput &gdi, DrawInfo &drawInfoOut) {
drawInfoOut.maxCommonControl = gdi.getSelectedItem("MaxCommonControl").first;
drawInfoOut.maxVacancy=gdi.getTextNo("VacancesMax");
drawInfoOut.minVacancy=gdi.getTextNo("VacancesMin");
drawInfoOut.vacancyFactor = 0.01*_wtof(gdi.getText("Vacances").c_str());
drawInfoOut.extraFactor = 0.01*_wtof(gdi.getText("Extra").c_str());
drawInfoOut.baseInterval=convertAbsoluteTimeMS(gdi.getText("BaseInterval"));
drawInfoOut.allowNeighbourSameCourse = gdi.isChecked("AllowNeighbours");
drawInfoOut.coursesTogether = gdi.isChecked("CoursesTogether");
drawInfoOut.minClassInterval = convertAbsoluteTimeMS(gdi.getText("MinInterval"));
drawInfoOut.maxClassInterval = convertAbsoluteTimeMS(gdi.getText("MaxInterval"));
drawInfoOut.nFields = gdi.getTextNo("nFields");
drawInfoOut.firstStart = oe->getRelativeTime(gdi.getText("FirstStart"));
}
void TabClass::writeDrawInfo(gdioutput &gdi, const DrawInfo &drawInfoIn) {
gdi.selectItemByData("MaxCommonControl", drawInfoIn.maxCommonControl);
gdi.setText("VacancesMax", drawInfoIn.maxVacancy);
gdi.setText("VacancesMin", drawInfoIn.minVacancy);
gdi.setText("Vacances", itow(int(drawInfoIn.vacancyFactor *100.0)) + L"%");
gdi.setText("Extra", itow(int(drawInfoIn.extraFactor * 100.0) ) + L"%");
gdi.setText("BaseInterval", formatTime(drawInfoIn.baseInterval));
gdi.check("AllowNeighbours", drawInfoIn.allowNeighbourSameCourse);
gdi.check("CoursesTogether", drawInfoIn.coursesTogether);
gdi.setText("MinInterval", formatTime(drawInfoIn.minClassInterval));
gdi.setText("MaxInterval", formatTime(drawInfoIn.maxClassInterval));
gdi.setText("nFields", drawInfoIn.nFields);
gdi.setText("FirstStart", oe->getAbsTime(drawInfoIn.firstStart));
}
void TabClass::setLockForkingState(gdioutput &gdi, bool poolState, bool lockState) {
if (gdi.hasField("DefineForking"))
gdi.setInputStatus("DefineForking", !lockState && !poolState);
if (gdi.hasField("LockForking"))
gdi.setInputStatus("LockForking", !poolState);
int legno = 0;
while (gdi.hasField("@Course" + itos(legno))) {
gdi.setInputStatus("@Course" + itos(legno++), !lockState || poolState);
}
}
bool TabClass::warnDrawStartTime(gdioutput &gdi, const wstring &firstStart) {
int st = oe->getRelativeTime(firstStart);
return warnDrawStartTime(gdi, st);
}
bool TabClass::warnDrawStartTime(gdioutput &gdi, int time) {
if (!hasWarnedStartTime && time > 3600 * 8 && !oe->useLongTimes()) {
bool res = gdi.ask(L"warn:latestarttime#" + itow(time/3600));
if (res)
hasWarnedStartTime = true;
return !res;
}
return false;
}
void TabClass::clearPage(gdioutput &gdi, bool autoRefresh) {
gdi.clearPage(autoRefresh);
gdi.setData("ClassPageLoaded", 1);
}