所以我已经被这个问题困扰了 3 周了,我终其一生都无法解决这个问题。 我想做的是使用表格获得这种输出/表示。
http://www.esl-world.net/masters/season6/hanover/sc2/playoffs/rankings/
这是踢球比赛的分组系统。 所以我的模型看起来像这样
public class Match{
public int id {get;set;}
public int teamid1 {get;set;}
public int teamid2 {get;set;}
public int roundnumber {get;set;}
public int winner {get;set;}
}
所以,我现在正在做的是先遍历各个回合,说,有几个回合我会这样做
for(int r = 1; r < bracketRounds; r++){
for(m = 1; m < roundMatches +1; m++){
matchGroup = "<tr><td>" + team1 + "</td></tr>"
+ "<tr><td>vs</td></tr>"
+ "<tr><td>" + team2 + "</td></tr>";
}
}
但是这只会生成一个包含 1 个列的表格来显示所有匹配项。想知道是否有人可以帮助/指出我应该如何处理这个问题的正确方向,以便我可以将后续行插入到第一行的右侧,这样它就会有一个像输出一样的括号。
谢谢!
最佳答案
这是我的尝试。我已经测试了 2、3 和 4 轮锦标赛的代码。此处显示了 2 轮和 3 轮锦标赛的输出:
我使用了您提供的相同模型来定义 Match
.我添加了一个 Tournament
生成测试数据的类。
Match.cs - 包含模型的类
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace tournament
{
public class Match
{
public int id { get; set; }
public int teamid1 { get; set; }
public int teamid2 { get; set; }
public int roundnumber { get; set; }
public int winner { get; set; }
public Match(int id, int teamid1, int teamid2, int roundnumber, int winner)
{
this.id = id;
this.teamid1 = teamid1;
this.teamid2 = teamid2;
this.roundnumber = roundnumber;
this.winner = winner;
}
}
public class Tournament
{
public SortedList<int, SortedList<int, Match>> TournamentRoundMatches { get; private set; }
public Match ThirdPlaceMatch { get; private set; }
public Tournament(int rounds)
{
this.TournamentRoundMatches = new SortedList<int, SortedList<int, Match>>();
this.GenerateTournamentResults(rounds);
if (rounds > 1)
{
this.GenerateThirdPlaceResult(rounds);
}
}
public void AddMatch(Match m)
{
if (this.TournamentRoundMatches.ContainsKey(m.roundnumber))
{
if (!this.TournamentRoundMatches[m.roundnumber].ContainsKey(m.id))
{
this.TournamentRoundMatches[m.roundnumber].Add(m.id, m);
}
}
else
{
this.TournamentRoundMatches.Add(m.roundnumber, new SortedList<int, Match>());
this.TournamentRoundMatches[m.roundnumber].Add(m.id, m);
}
}
private void GenerateTournamentResults(int rounds)
{
Random WinnerRandomizer = new Random();
for (int round = 1, match_id = 1; round <= rounds; round++)
{
int matches_in_round = 1 << (rounds - round);
for (int round_match = 1; round_match <= matches_in_round; round_match++, match_id++)
{
int team1_id;
int team2_id;
int winner;
if (round == 1)
{
team1_id = (match_id * 2) - 1;
team2_id = (match_id * 2);
}
else
{
int match1 = (match_id - (matches_in_round * 2) + (round_match - 1));
int match2 = match1 + 1;
team1_id = this.TournamentRoundMatches[round - 1][match1].winner;
team2_id = this.TournamentRoundMatches[round - 1][match2].winner;
}
winner = (WinnerRandomizer.Next(1, 3) == 1) ? team1_id : team2_id;
this.AddMatch(new Match(match_id, team1_id, team2_id, round, winner));
}
}
}
private void GenerateThirdPlaceResult(int rounds)
{
Random WinnerRandomizer = new Random();
int semifinal_matchid1 = this.TournamentRoundMatches[rounds - 1].Keys.ElementAt(0);
int semifinal_matchid2 = this.TournamentRoundMatches[rounds - 1].Keys.ElementAt(1);
Match semifinal_1 = this.TournamentRoundMatches[rounds - 1][semifinal_matchid1];
Match semifinal_2 = this.TournamentRoundMatches[rounds - 1][semifinal_matchid2];
int semifinal_loser1 = (semifinal_1.winner == semifinal_1.teamid1) ? semifinal_1.teamid2 : semifinal_1.teamid1;
int semifinal_loser2 = (semifinal_2.winner == semifinal_2.teamid1) ? semifinal_2.teamid2 : semifinal_2.teamid1;
int third_place_winner = (WinnerRandomizer.Next(1, 3) == 1) ? semifinal_loser1 : semifinal_loser2;
this.ThirdPlaceMatch = new Match((1 << rounds) + 1, semifinal_loser1, semifinal_loser2, 1, third_place_winner);
}
}
}
我使用静态方法动态生成了原始 HTML GenerateHTMLResultsTable
.这是仅使用 <table>
完成的不需要 <div>
block 。
Program.cs - 初始化测试数据并生成 HTML 的静态程序类
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;
namespace tournament
{
class Program
{
static string GenerateHTMLResultsTable(Tournament tournament)
{
int match_white_span;
int match_span;
int position_in_match_span;
int column_stagger_offset;
int effective_row;
int col_match_num;
int cumulative_matches;
int effective_match_id;
int rounds = tournament.TournamentRoundMatches.Count;
int teams = 1 << rounds;
int max_rows = teams << 1;
StringBuilder HTMLTable = new StringBuilder();
HTMLTable.AppendLine("<style type=\"text/css\">");
HTMLTable.AppendLine(" .thd {background: rgb(220,220,220); font: bold 10pt Arial; text-align: center;}");
HTMLTable.AppendLine(" .team {color: white; background: rgb(100,100,100); font: bold 10pt Arial; border-right: solid 2px black;}");
HTMLTable.AppendLine(" .winner {color: white; background: rgb(60,60,60); font: bold 10pt Arial;}");
HTMLTable.AppendLine(" .vs {font: bold 7pt Arial; border-right: solid 2px black;}");
HTMLTable.AppendLine(" td, th {padding: 3px 15px; border-right: dotted 2px rgb(200,200,200); text-align: right;}");
HTMLTable.AppendLine(" h1 {font: bold 14pt Arial; margin-top: 24pt;}");
HTMLTable.AppendLine("</style>");
HTMLTable.AppendLine("<h1>Tournament Results</h1>");
HTMLTable.AppendLine("<table border=\"0\" cellspacing=\"0\">");
for (int row = 0; row <= max_rows; row++)
{
cumulative_matches = 0;
HTMLTable.AppendLine(" <tr>");
for (int col = 1; col <= rounds + 1; col++)
{
match_span = 1 << (col + 1);
match_white_span = (1 << col) - 1;
column_stagger_offset = match_white_span >> 1;
if (row == 0)
{
if (col <= rounds)
{
HTMLTable.AppendLine(" <th class=\"thd\">Round " + col + "</th>");
}
else
{
HTMLTable.AppendLine(" <th class=\"thd\">Winner</th>");
}
}
else if (row == 1)
{
HTMLTable.AppendLine(" <td class=\"white_span\" rowspan=\"" + (match_white_span - column_stagger_offset) + "\"> </td>");
}
else
{
effective_row = row + column_stagger_offset;
if (col <= rounds)
{
position_in_match_span = effective_row % match_span;
position_in_match_span = (position_in_match_span == 0) ? match_span : position_in_match_span;
col_match_num = (effective_row / match_span) + ((position_in_match_span < match_span) ? 1 : 0);
effective_match_id = cumulative_matches + col_match_num;
if ((position_in_match_span == 1) && (effective_row % match_span == position_in_match_span))
{
HTMLTable.AppendLine(" <td class=\"white_span\" rowspan=\"" + match_white_span + "\"> </td>");
}
else if ((position_in_match_span == (match_span >> 1)) && (effective_row % match_span == position_in_match_span))
{
HTMLTable.AppendLine(" <td class=\"team\">Team " + tournament.TournamentRoundMatches[col][effective_match_id].teamid1 + "</td>");
}
else if ((position_in_match_span == ((match_span >> 1) + 1)) && (effective_row % match_span == position_in_match_span))
{
HTMLTable.AppendLine(" <td class=\"vs\" rowspan=\"" + match_white_span + "\">VS</td>");
}
else if ((position_in_match_span == match_span) && (effective_row % match_span == 0))
{
HTMLTable.AppendLine(" <td class=\"team\">Team " + tournament.TournamentRoundMatches[col][effective_match_id].teamid2 + "</td>");
}
}
else
{
if (row == column_stagger_offset + 2)
{
HTMLTable.AppendLine(" <td class=\"winner\">Team " + tournament.TournamentRoundMatches[rounds][cumulative_matches].winner + "</td>");
}
else if (row == column_stagger_offset + 3)
{
HTMLTable.AppendLine(" <td class=\"white_span\" rowspan=\"" + (match_white_span - column_stagger_offset) + "\"> </td>");
}
}
}
if (col <= rounds)
{
cumulative_matches += tournament.TournamentRoundMatches[col].Count;
}
}
HTMLTable.AppendLine(" </tr>");
}
HTMLTable.AppendLine("</table>");
HTMLTable.AppendLine("<h1>Third Place Results</h1>");
HTMLTable.AppendLine("<table border=\"0\" cellspacing=\"0\">");
HTMLTable.AppendLine(" <tr>");
HTMLTable.AppendLine(" <th class=\"thd\">Round 1</th>");
HTMLTable.AppendLine(" <th class=\"thd\">Third Place</th>");
HTMLTable.AppendLine(" </tr>");
HTMLTable.AppendLine(" <tr>");
HTMLTable.AppendLine(" <td class=\"white_span\"> </td>");
HTMLTable.AppendLine(" <td class=\"white_span\" rowspan=\"2\"> </td>");
HTMLTable.AppendLine(" </tr>");
HTMLTable.AppendLine(" <tr>");
HTMLTable.AppendLine(" <td class=\"team\">Team " + tournament.ThirdPlaceMatch.teamid1 + "</td>");
HTMLTable.AppendLine(" </tr>");
HTMLTable.AppendLine(" <tr>");
HTMLTable.AppendLine(" <td class=\"vs\">VS</td>");
HTMLTable.AppendLine(" <td class=\"winner\">Team " + tournament.ThirdPlaceMatch.winner + "</td>");
HTMLTable.AppendLine(" </tr>");
HTMLTable.AppendLine(" <tr>");
HTMLTable.AppendLine(" <td class=\"team\">Team " + tournament.ThirdPlaceMatch.teamid2 + "</td>");
HTMLTable.AppendLine(" <td class=\"white_span\"> </td>");
HTMLTable.AppendLine(" </tr>");
HTMLTable.AppendLine("</table>");
return HTMLTable.ToString();
}
static void Main(string[] args)
{
Tournament Test3RoundTournament = new Tournament(3);
Tournament Test2RoundTournament = new Tournament(2);
File.WriteAllText(@"C:\Tournament\results.html", GenerateHTMLResultsTable(Test2RoundTournament));
File.WriteAllText(@"C:\Tournament\results.html", GenerateHTMLResultsTable(Test3RoundTournament));
Console.ReadLine();
}
}
}
更新
用于生成 HTML 表格的参数说明
如您所见,column_stagger_offset
是每列向上移动以使它们按预期方式对齐的量。 effective_row
如果没有垂直移动,则基本上是特定表格单元格所在的位置。知道effective_row
和 position_in_match_span
帮助确定需要在该特定单元格中显示的内容(空白、team1、team2 或 vs.)。
如您所见,我一次一行地遍历列。考虑到 HTML 表格也是以这种方式构建的,即创建行、添加单元格……创建行、添加单元格……等等,这似乎是最自然的。
关于c# 如何生成锦标赛分组 HTML 表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9732347/