Como criar dias da semana recorrentes como colunas em um pivô?

8

Sou iniciante em programação e bancos de dados e agradeceria por alguma ajuda no cenário a seguir.

Eu uso PHP com SQL Server. Estou criando um sistema de presença de funcionários e gostaria de criar uma tabela (dinâmica) com meses como linhas e o nome de todos os dias da semana como colunas (para um ano específico). Os valores nas células serão o número do dia (1, 2, 3 ... 31).

A cor de plano de fundo da célula (já existe como coluna da tabela) declara o tipo de licença dos funcionários. A tabela tem as seguintes colunas: employee_id, leave_date, leave_type, leave_type_color.

Eu quero alcançar um resultado como abaixo:

insira a descrição da imagem aqui

Obrigado.

Mike T
fonte
Obrigado por um problema interessante! Não estou entusiasmado com a mistura de dados e apresentação, mas em alguns casos ter toda a lógica em um só lugar pode ser prático.
Aaron Bertrand

Respostas:

11

A parte mais complexa disso é apenas criar o calendário nesse formato. Girar e cercar com HTML é bastante fácil. Primeiro, vamos começar com isso, sua tabela de funcionários com datas de folga. leave_typenão parecia relevante para o problema em questão.

CREATE TABLE dbo.EmpLeave
(
  EmployeeID int,
  leave_date date,
  leave_type_color char(6)
);

INSERT dbo.EmpLeave(EmployeeID,leave_date,leave_type_color)
  VALUES(1,'2018-01-02','7777cc'),(1,'2018-04-01','ffffac');

O procedimento que criei se parece com isso (e aviso: assume @@DATEFIRST = 7):

CREATE PROCEDURE dbo.BuildLeaveHTMLTable
  @EmployeeID int,
  @Year smallint = NULL
AS
BEGIN
  SET NOCOUNT ON;
  SET @Year = COALESCE(@Year, DATEPART(YEAR, GETDATE()));
  DECLARE @FirstDay date = DATEADD(YEAR, @Year-1900, 0);

  ;WITH Numbers AS ( -- 366 possible days (leap year)
    SELECT n = 1 UNION ALL SELECT n + 1 FROM Numbers WHERE n <= 365
  ),
  Calendar AS ( -- a year's worth of dates and dateparts 
    SELECT [Date] = d,
      MonthStart = DATEADD(DAY, 1-DAY(d),d),
      Y  = CONVERT(smallint, DATEPART(YEAR,   d)),
      M  = CONVERT(tinyint,  DATEPART(MONTH,  d)),
      D  = CONVERT(tinyint,  DATEPART(DAY,    d)),
      WY = CONVERT(tinyint,  DATEPART(WEEK,   d)),
      DW = CONVERT(tinyint,  DATEPART(WEEKDAY,d))
    FROM
    (
      SELECT d = CONVERT(date,DATEADD(DAY, n-1, @FirstDay)) FROM Numbers
    ) AS c WHERE YEAR(d) = @Year -- in case it's not a leap year
  ),
  BaseSlots AS ( -- base set of 37 ints 
   -- month can be spread across 6 weeks, but no more than 2 days in 6th week
    SELECT TOP (37) slot = n FROM Numbers ORDER BY n
  ),
  Months AS ( -- base set of 12 ints
    SELECT TOP (12) m = slot FROM BaseSlots ORDER BY slot
  ),
  SlotAlignment AS ( -- align days of week to slot numbers
    -- this is the most cryptic part of this solution
    -- determines which set of 7 slots, and which slot 
    -- exactly, a given date will appear under
    SELECT c.*, slot = DW+(c.WY+1-DATEPART(WEEK,c.MonthStart)-1)*7
      FROM Calendar AS c 
      INNER JOIN Months AS m ON c.M = m.m
  ),
  SlotMatrix AS ( -- extrapolate actual dates to 37 x 12 matrix
    SELECT m.m, s.slot, sa.[Date] 
      FROM BaseSlots AS s 
      CROSS JOIN Months AS m
      LEFT OUTER JOIN SlotAlignment AS sa
      ON sa.m = m.m AND sa.slot = s.slot
  ),
  FinalHTML AS ( -- build some HTML!
    SELECT m = '<!-- ' + RIGHT('0' + RTRIM(m), 2) + ' -->', 
      slot, cell = CASE WHEN slot = 1 THEN '<tr><th>' 
        + COALESCE(DATENAME(MONTH,DATEADD(MONTH, m-1, 0)),'') 
        + '</th>' ELSE '' END + '<td' + COALESCE(' bgcolor=#' 
        + RIGHT(CONVERT(varchar(10),CONVERT(varbinary(8), el.leave_type_color),1),6),
          CASE WHEN DATEPART(WEEKDAY, [Date]) IN (1,7) 
          THEN ' bgcolor=#cccccc' ELSE '' END)
        + '>' + COALESCE(RTRIM(DATEPART(DAY,[Date])), '&nbsp;')
        + '</td>' + CASE WHEN slot = 37 THEN '</tr>' ELSE '' END
      FROM SlotMatrix AS q LEFT OUTER JOIN dbo.EmpLeave AS el
      ON q.Date = el.leave_date
      AND el.EmployeeID = @EmployeeID
  ) -- now turn it sideways
  SELECT m = '<!-- 00 -->', 
    [1]  = '<tr><th>Month</th><th>S</th>',    [2]  = '<th>M</th>', 
    [3]  = '<th>T</th>', [4]  = '<th>W</th>', [5]  = '<th>T</th>', 
    [6]  = '<th>F</th>', [7]  = '<th>S</th>', [8]  = '<th>S</th>', 
    [9]  = '<th>M</th>', [10] = '<th>T</th>', [11] = '<th>W</th>',
    [12] = '<th>T</th>', [13] = '<th>F</th>', [14] = '<th>S</th>', 
    [15] = '<th>S</th>', [16] = '<th>M</th>', [17] = '<th>T</th>',
    [18] = '<th>W</th>', [19] = '<th>T</th>', [20] = '<th>F</th>', 
    [21] = '<th>S</th>', [22] = '<th>S</th>', [23] = '<th>M</th>',
    [24] = '<th>T</th>', [25] = '<th>W</th>', [26] = '<th>T</th>', 
    [27] = '<th>F</th>', [28] = '<th>S</th>', [29] = '<th>S</th>', 
    [30] = '<th>M</th>', [31] = '<th>T</th>', [32] = '<th>W</th>', 
    [33] = '<th>T</th>', [34] = '<th>F</th>', [35] = '<th>S</th>',
    [36] = '<th>S</th>', [37] = '<th>M</th>'
  UNION ALL
  (
    SELECT * FROM FinalHTML PIVOT (MAX(cell) FOR slot IN 
    (
     [1], [2], [3], [4], [5], [6], [7], [8], [9], [10],[11],[12],[13],[14],
     [15],[16],[17],[18],[19],[20],[21],[22],[23],[24],[25],[26],[27],[28],
     [29],[30],[31],[32],[33],[34],[35],[36],[37]
    )) AS p
  )
  ORDER BY m OPTION (MAXRECURSION 366);
END
GO

Resultados desta chamada:

EXEC dbo.BuildLeaveHTMLTable @EmployeeID = 1;

É assim (parei na coluna do sétimo dia):

insira a descrição da imagem aqui

Você precisará adicionar o <table>/ </table>wrapper, mas aqui está a aparência da saída quando colocada entre elas e salva como HTML (e, é claro, você pode aprimorá-la ainda mais com CSS):

! [insira a descrição da imagem aqui

Quando a licença cai em um fim de semana, a cor da licença supera a cor do fim de semana, mas isso é fácil de ajustar. Mude isso:

  + COALESCE(' bgcolor=#' + RTRIM(el.leave_type_color),
      CASE WHEN DATEPART(WEEKDAY, [Date]) IN (1,7) 
      THEN ' bgcolor=#cccccc' ELSE '' END)

Para isso:

  + CASE WHEN DATEPART(WEEKDAY, [Date]) IN (1,7) 
      THEN ' bgcolor=#cccccc' ELSE COALESCE(' bgcolor=#' 
      + RTRIM(el.leave_type_color), '') END

Para converter uma cor no formato decimal (como 65280) em seu equivalente RGB ( 00FF00), você precisa fazer um monte de manipulação. Eu consideraria armazená-lo como hexadecimal RGB em primeiro lugar, mas atualizei a solução aqui com algo semelhante a este:

SELECT RIGHT(CONVERT(varchar(10),CONVERT(varbinary(8), 65280),1),6);
Aaron Bertrand
fonte
Sim. O que Aaron disse.
Rob Farley
2
Você é tão estranho.
Erik Darling
Obrigado pela ajuda.Eu recebo o erro: Falha na conversão ao converter o valor varchar '>' para o tipo de dados int.
Mike T
@ MikeT Esse código foi totalmente testado, o que você mudou? A leave_type_colorcoluna é numérica?
Aaron Bertrand
1) "DECLARE @return_value int" desempenha um papel quando executo o procedimento no SQL 2016? 2) Alterei alguns nomes de coluna porque a tabela de saída é uma junção de 2 outras tabelas.leave_type_color é inteiro.
Mike T
1

Comece considerando o que você deseja ter como colunas, e isso é basicamente "Semana 1 Dia 1 (Dom)", "Semana 1 Dia 2 (Seg)", até "Semana 6 Dia 7 (Sábado)". Essencialmente, dia 1-42. 1º de janeiro é então "Semana 1 Dia 2" de janeiro. Vou ligar para o WeekPlusDay por enquanto.

Para descobrir onde cada um começa, considere a parte do dia da semana da data.

Seu conjunto de dados precisa incluir apenas esse valor "WeekPlusDay" e você exibirá o DayOfMonth.

Rob Farley
fonte