declare
@DateBegin datetime,
@DateEnd datetime,
@GLCode varchar(250),
@Service varchar(250),
@Location varchar(250),
@Employee int;
set @DateBegin = :DateBegin;
set @DateEnd = :DateEnd;
set @Location = :Location;
set @Service = :Service;
set @GLCode = :GLCode;
set @Employee = :Employee;
declare @header table (sort int, description varchar(50));
insert into @header (sort, description)
select 100, 'Location'
union all
select 200, 'Date'
union all
select 300, 'Sub-Total Sales'
union all
select 400, '*Services'
union all
select 500, '*GL Codes (Sales)'
union all
select 600, '*GL Codes (Discounts)'
union all
select 700, 'Discounts'
union all
select 800, 'Taxes'
union all
select 999, 'Total Sales';
print '@dates';
declare @dates table (date datetime, sort int);
insert into @dates
select dateadd(day, number, @DateBegin), number + 1
from master..spt_values
where type = 'P'
and number between 0 and datediff(day, @DateBegin, @DateEnd)
union all
select null, 99999;
print '@services';
declare @services table (code varchar(50), sort int);
insert into @services (code, sort)
select
s.HistoryCode,
row_number() over (order by min(s.HistorySorting), s.HistoryCode)
from (
select distinct
s.HistoryCode,
s.HistorySorting
from dbo.fnSplit(@Service) f
inner join Service s on s.HistoryCode = ltrim(f.value)
) s
group by s.HistoryCode
;
print '@glcodes';
declare @glcodes table (id int, code varchar(50), name varchar(50), coef int, sort int);
insert into @glcodes
select
p.ID, p.Code, p.Name,
case p.PaidOut when 1 then -1 else 1 end,
row_number() over (order by p.Code)
from dbo.fnSplit(@GLCode) s
inner join PaidOutCode p on ltrim(s.value) = p.Code;
print '@locations';
declare @locations table (id int, code varchar(10), name varchar(50), sort int);
insert into @locations
select
ID,
code,
Name,
row_number() over (order by Type, code)
from (
select distinct
l.ID,
l.Type,
case l.Type when 0 then 'O-' else 'F-' end + right('0000' + l.Code, 4) as code,
l.Name
from dbo.fnSplit(@Location) s
inner join Location l
on s.Value = 'o' and l.Type = 0 and l.Active = 1 and cast(l.Code as int) < 9000
or s.Value = 'f' and l.Type = 1 and l.Active = 1 and cast(l.Code as int) < 9000
or s.Value not in ('o', 'f') and cast(l.ID as varchar) = ltrim(s.Value)
inner join (
select [Object Id2] as Location
from Availability
where [Object Identification1] = 4
and [Object Identification2] = 8
and [Object Id1] = @Employee
) a on a.Location = l.ID
) s
union all
select null, null, null, 0
union all
select null, null, null, 99999;
print '@orders';
declare @orders table (
ID int,
Location int,
Sale money,
Discount money,
Tax money,
Date datetime
);
insert into @orders (ID, Location, Sale, Discount, Tax, Date)
select o.ID, o.Location, o.Sale, o.Discount, o.Tax, dateadd(day, datediff(day, 0, DateTimeBegin), 0)
from [Order] o
inner join @locations l on o.Location = l.ID
/*inner join @dates d on o.DateTimeBegin >= d.Date and o.DateTimeBegin < d.Date + 1*/
where o.Status = 0
and o.DateTimeBegin >= @DateBegin
and o.DateTimeBegin < @DateEnd + 1;
print 'TempList';
delete from TempList where spid = @@spid;
insert into TempList (spid, val)
select @@spid, o.ID
from @orders o
inner join [Order Item] oi on o.ID = oi.[Order]
inner join Discount d on oi.[Object Identification] = 2 and oi.[Object ID] = d.ID
inner join @glcodes g on d.GLCode = g.ID and d.ReportAsPaidOut = 1;
print '#ReportSalesAnalysis_CalculatedOrders';
if object_id('tempdb..#ReportSalesAnalysis_CalculatedOrders') is not null drop table #ReportSalesAnalysis_CalculatedOrders;
select *
into #ReportSalesAnalysis_CalculatedOrders
from dbo.fnCalculateMoneyMultiple(0)
print '@glcodesdiscounts';
declare @glcodesdiscounts table (glcodesort int, locationsort int, datesort int, Amount money);
insert into @glcodesdiscounts (glcodesort, locationsort, datesort, Amount)
select
s.glcodesort,
s.locationsort,
d.sort,
sum(s.Amount)
from (
select
glcodesort = g.sort,
locationsort = l.sort,
Date = dateadd(day, datediff(day, 0, o.DateTimeBegin), 0),
Amount = round(case oi.Quantity
when 0 then oi.Price * (c.SubTotalPercentable + d.UpdateTax * c.SalesTaxPercentable) / 100
else oi.Price
end, 2)
from #ReportSalesAnalysis_CalculatedOrders c
inner join [Order] o on c.[Order] = o.ID
inner join @locations l on o.Location = l.ID
inner join [Order Item] oi on c.[Order] = oi.[Order] and oi.[Object Identification] = 2
inner join Discount d on oi.[Object ID] = d.ID and d.ReportAsPaidOut = 1
inner join @glcodes g on d.GLCode = g.ID and d.ReportAsPaidOut = 1
) s
inner join @dates d on s.Date = d.date
group by
s.glcodesort,
s.locationsort,
d.sort;
insert into @orders (Location, Date, Discount)
select l.ID, d.Date, -g.Amount
from @glcodesdiscounts g
inner join @locations l on g.locationsort = l.sort
inner join @dates d on g.datesort = d.sort;
insert into @glcodesdiscounts (glcodesort, locationsort, datesort, Amount)
select glcodesort, locationsort, 99999, sum(Amount) from @glcodesdiscounts group by glcodesort, locationsort
union all
select glcodesort, 99999, 99999, sum(Amount) from @glcodesdiscounts group by glcodesort
union all
select 999, locationsort, datesort, sum(Amount) from @glcodesdiscounts group by locationsort, datesort
union all
select 999, locationsort, 99999, sum(Amount) from @glcodesdiscounts group by locationsort
union all
select 999, 99999, 99999, sum(Amount) from @glcodesdiscounts group by case datesort when 0 then 0 end;
print '@glcodessales';
declare @glcodessales table (glcodesort int, locationsort int, datesort int, Amount money);
insert into @glcodessales (glcodesort, locationsort, datesort, Amount)
select
s.glcodesort,
s.locationsort,
d.sort,
sum(Amount)
from (
select
glcodesort = g.sort,
locationsort = l.sort,
Date = dateadd(day, datediff(day, 0, p.DateTime), 0),
Amount = p.Amount * g.coef
from PaidOut p
inner join @glcodes g on p.PaidOutCode = g.id
inner join @locations l on p.Location = l.ID
where p.DateTime >= @DateBegin
and p.DateTime < @DateEnd + 1
) s
inner join @dates d on s.Date = d.date
group by
s.glcodesort,
s.locationsort,
d.sort;
insert into @glcodessales (glcodesort, locationsort, datesort, Amount)
select glcodesort, locationsort, 99999, sum(Amount) from @glcodessales group by glcodesort, locationsort
union all
select glcodesort, 99999, 99999, sum(Amount) from @glcodessales group by glcodesort
union all
select 999, locationsort, datesort, sum(Amount) from @glcodessales group by locationsort, datesort
union all
select 999, locationsort, 99999, sum(Amount) from @glcodessales group by locationsort
union all
select 999, 99999, 99999, sum(Amount) from @glcodessales group by case datesort when 0 then 0 end;
print '@servicemoney';
declare @servicemoney table (servicesort int, locationsort int, datesort int, Amount money);
insert into @servicemoney (servicesort, locationsort, datesort, Amount)
select
s.sort,
l.sort,
d.sort,
Amount = sum(ot.Price)
from [OIL STOP REPORT].[OIL STOP REPORT].dbo.OperativeTotals ot
inner join [OIL STOP REPORT].[OIL STOP REPORT].dbo.OperativeTotalsItemsDescription otid
on ot.Object = otid.Object and ot.ObjectID = otid.ObjectID
inner join @services s on otid.HistoryCode = s.code
inner join @dates d on ot.Date = d.date
inner join @locations l on ot.Location = l.id
group by
s.sort,
l.sort,
d.sort;
insert into @orders (Location, Date, Sale)
select l.ID, d.Date, -s.Amount
from @servicemoney s
inner join @locations l on s.locationsort = l.sort
inner join @dates d on s.datesort = d.sort;
insert into @servicemoney (servicesort, locationsort, datesort, Amount)
select servicesort, locationsort, 99999, sum(Amount) from @servicemoney group by servicesort, locationsort
union all
select servicesort, 99999, 99999, sum(Amount) from @servicemoney group by servicesort
union all
select 999, locationsort, datesort, sum(Amount) from @servicemoney group by locationsort, datesort
union all
select 999, locationsort, 99999, sum(Amount) from @servicemoney group by locationsort
union all
select 999, 99999, 99999, sum(Amount) from @servicemoney group by case datesort when 0 then 0 end;
print '@ordermoney';
declare @ordermoney table (locationsort int, datesort int, Sale money, Discount money, Tax money);
insert into @ordermoney (locationsort, datesort, Sale, Discount, Tax)
select
l.sort,
d.sort,
Sale = sum(Sale),
Discount = sum(Discount),
Tax = sum(Tax)
from @orders o
inner join @locations l on o.Location = l.ID
inner join @dates d on o.Date = d.date
group by
l.sort,
d.sort;
insert into @ordermoney (locationsort, datesort, Sale, Discount, Tax)
select locationsort, 99999, sum(Sale), sum(Discount), sum(Tax) from @ordermoney group by locationsort
union all
select 99999, 99999, sum(Sale), sum(Discount), sum(Tax) from @ordermoney
union all
select 0, 99999, null, null, null;
if not exists (select * from @servicemoney)
delete from @header where sort = 400;
if not exists (select * from @glcodessales)
delete from @header where sort = 500;
if not exists (select * from @glcodesdiscounts)
delete from @header where sort = 600;
if object_id('tempdb..#ReportSalesAnalysis') is not null drop table #ReportSalesAnalysis;
select
ld.locationsort,
ld.datesort,
columnsort = h.sort + case h.sort
when 400 then ss.sort
when 500 then ggs.sort
when 600 then ggd.sort
else 0
end,
Amount =
case h.sort
when 300 then isnull(o.Sale, 0)
when 400 then isnull(s.Amount, 0)
when 500 then isnull(gs.Amount, 0)
when 600 then -isnull(gd.Amount, 0)
when 700 then -isnull(o.Discount, 0)
when 800 then isnull(o.Tax, 0)
when 999 then isnull(o.Sale, 0) + isnull(s.Amount, 0) + isnull(gs.Amount, 0) - isnull(gd.Amount, 0) - isnull(o.Discount, 0) + isnull(o.Tax, 0)
end,
Caption =
case h.sort
when 100 then case ll.sort when 0 then h.description else case dd.sort when 99999 then 'Total' else ll.code end end
when 200 then case ll.sort when 0 then h.description else convert(varchar, dd.date, 101) end
when 400 then ss.code
when 500 then ggs.code + ' ' + ggs.name
when 600 then ggd.code + ' (Discounts)'
else h.description
end
into #ReportSalesAnalysis
from (select distinct locationsort from @ordermoney) l
inner join (select distinct locationsort, datesort from @ordermoney) ld on l.locationsort = ld.locationsort
inner join @locations ll on l.locationsort = ll.sort
inner join @dates dd on ld.datesort = dd.sort
cross join @header h
left join @ordermoney o on h.sort in (300, 700, 800, 999)
and ld.locationsort = o.locationsort
and ld.datesort = o.datesort
left join (select distinct t.*, m.servicesort from @servicemoney m left join @services t on m.servicesort = t.sort) ss
on h.sort = ss.servicesort or (h.sort = 400 and ss.servicesort < 100)
left join @servicemoney s on ss.servicesort = s.servicesort
and ld.locationsort = s.locationsort -- this line produces the error
-- the error is:
-- The multi-part identifier "ld.locationsort" could not be bound
and ld.datesort = s.datesort
left join (select distinct t.*, m.glcodesort from @glcodessales m left join @glcodes t on m.glcodesort = t.sort) ggs
on h.sort = ggs.glcodesort or (h.sort = 500 and ggs.glcodesort < 100)
left join @glcodessales gs on ggs.glcodesort = gs.glcodesort
and ld.locationsort = gs.locationsort
and ld.datesort = gs.datesort
left join (select distinct t.*, m.glcodesort from @glcodesdiscounts m left join @glcodes t on m.glcodesort = t.sort) ggd
on h.sort = ggd.glcodesort or (h.sort = 600 and ggd.glcodesort < 100)
left join @glcodesdiscounts gd on ggd.glcodesort = gd.glcodesort
and ld.locationsort = gd.locationsort
and ld.datesort = gd.datesort;
select
*, (select count(distinct columnsort) from #ReportSalesAnalysis) as ColumnCnt
from #ReportSalesAnalysis
order by
ld.locationsort,
ld.datesort,
columnsort;
declare
  @DateBegin datetime,
  @DateEnd datetime,
  @GLCode varchar(250),
  @Service varchar(250),
  @Location varchar(250),
  @Employee int;

set @DateBegin = :DateBegin;
set @DateEnd = :DateEnd;
set @Location = :Location;
set @Service = :Service;
set @GLCode = :GLCode;
set @Employee = :Employee;

declare @header table (sort int, description varchar(50));
insert into @header (sort, description)
select 100, 'Location'
union all
select 200, 'Date'
union all
select 300, 'Sub-Total Sales'
union all
select 400, '*Services'
union all
select 500, '*GL Codes (Sales)'
union all
select 600, '*GL Codes (Discounts)'
union all
select 700, 'Discounts'
union all
select 800, 'Taxes'
union all
select 999, 'Total Sales';

print '@dates';
declare @dates table (date datetime, sort int);
insert into @dates
select dateadd(day, number, @DateBegin), number + 1
from master..spt_values
where type = 'P'
  and number between 0 and datediff(day, @DateBegin, @DateEnd)
union all
select null, 99999;

print '@services';
declare @services table (code varchar(50), sort int);
insert into @services (code, sort)
select
  s.HistoryCode,
  row_number() over (order by min(s.HistorySorting), s.HistoryCode)
from (
  select distinct
    s.HistoryCode,
    s.HistorySorting
  from dbo.fnSplit(@Service) f
    inner join Service s on s.HistoryCode = ltrim(f.value)
) s
group by s.HistoryCode
;

print '@glcodes';
declare @glcodes table (id int, code varchar(50), name varchar(50), coef int, sort int);
insert into @glcodes
select
  p.ID, p.Code, p.Name,
  case p.PaidOut when 1 then -1 else 1 end,
  row_number() over (order by p.Code)
from dbo.fnSplit(@GLCode) s
  inner join PaidOutCode p on ltrim(s.value) = p.Code;

print '@locations';
declare @locations table (id int, code varchar(10), name varchar(50), sort int);
insert into @locations
select
  ID,
  code,
  Name,
  row_number() over (order by Type, code)
from (
  select distinct
    l.ID,
    l.Type,
    case l.Type when 0 then 'O-' else 'F-' end + right('0000' + l.Code, 4) as code,
    l.Name
  from dbo.fnSplit(@Location) s
    inner join Location l
      on s.Value = 'o' and l.Type = 0 and l.Active = 1 and cast(l.Code as int) < 9000
      or s.Value = 'f' and l.Type = 1 and l.Active = 1 and cast(l.Code as int) < 9000
      or s.Value not in ('o', 'f') and cast(l.ID as varchar) = ltrim(s.Value)
    inner join (
      select [Object Id2] as Location
      from Availability
      where [Object Identification1] = 4
        and [Object Identification2] = 8
        and [Object Id1] = @Employee
    ) a on a.Location = l.ID
) s
union all
select null, null, null, 0
union all
select null, null, null, 99999;

print '@orders';
declare @orders table (
  ID int,
  Location int,
  Sale money,
  Discount money,
  Tax money,
  Date datetime
);
insert into @orders (ID, Location, Sale, Discount, Tax, Date)
select o.ID, o.Location, o.Sale, o.Discount, o.Tax, dateadd(day, datediff(day, 0, DateTimeBegin), 0)
from [Order] o
  inner join @locations l on o.Location = l.ID
  /*inner join @dates d on o.DateTimeBegin >= d.Date and o.DateTimeBegin < d.Date + 1*/
where o.Status = 0
  and o.DateTimeBegin >= @DateBegin
  and o.DateTimeBegin < @DateEnd + 1;

print 'TempList';
delete from TempList where spid = @@spid;
insert into TempList (spid, val)
select @@spid, o.ID
from @orders o
  inner join [Order Item] oi on o.ID = oi.[Order]
  inner join Discount d on oi.[Object Identification] = 2 and oi.[Object ID] = d.ID
  inner join @glcodes g on d.GLCode = g.ID and d.ReportAsPaidOut = 1;

print '#ReportSalesAnalysis_CalculatedOrders';
if object_id('tempdb..#ReportSalesAnalysis_CalculatedOrders') is not null drop table #ReportSalesAnalysis_CalculatedOrders;
select *
into #ReportSalesAnalysis_CalculatedOrders
from dbo.fnCalculateMoneyMultiple(0)

print '@glcodesdiscounts';
declare @glcodesdiscounts table (glcodesort int, locationsort int, datesort int, Amount money);
insert into @glcodesdiscounts (glcodesort, locationsort, datesort, Amount)
select
  s.glcodesort,
  s.locationsort,
  d.sort,
  sum(s.Amount)
from (
  select
    glcodesort = g.sort,
    locationsort = l.sort,
    Date = dateadd(day, datediff(day, 0, o.DateTimeBegin), 0),
    Amount = round(case oi.Quantity
      when 0 then oi.Price * (c.SubTotalPercentable + d.UpdateTax * c.SalesTaxPercentable) / 100
      else oi.Price
    end, 2)
  from #ReportSalesAnalysis_CalculatedOrders c
    inner join [Order] o on c.[Order] = o.ID
    inner join @locations l on o.Location = l.ID
    inner join [Order Item] oi on c.[Order] = oi.[Order] and oi.[Object Identification] = 2
    inner join Discount d on oi.[Object ID] = d.ID and d.ReportAsPaidOut = 1
    inner join @glcodes g on d.GLCode = g.ID and d.ReportAsPaidOut = 1
) s
  inner join @dates d on s.Date = d.date
group by
  s.glcodesort,
  s.locationsort,
  d.sort;

insert into @orders (Location, Date, Discount)
select l.ID, d.Date, -g.Amount
from @glcodesdiscounts g
  inner join @locations l on g.locationsort = l.sort
  inner join @dates d on g.datesort = d.sort;

insert into @glcodesdiscounts (glcodesort, locationsort, datesort, Amount)
select glcodesort, locationsort, 99999, sum(Amount)    from @glcodesdiscounts group by glcodesort, locationsort
union all
select glcodesort, 99999,        99999, sum(Amount)    from @glcodesdiscounts group by glcodesort
union all
select 999,        locationsort, datesort, sum(Amount) from @glcodesdiscounts group by locationsort, datesort
union all
select 999,        locationsort, 99999,    sum(Amount) from @glcodesdiscounts group by locationsort
union all
select 999,        99999,        99999,    sum(Amount) from @glcodesdiscounts group by case datesort when 0 then 0 end;

print '@glcodessales';
declare @glcodessales table (glcodesort int, locationsort int, datesort int, Amount money);
insert into @glcodessales (glcodesort, locationsort, datesort, Amount)
select
  s.glcodesort,
  s.locationsort,
  d.sort,
  sum(Amount)
from (
  select
    glcodesort = g.sort,
    locationsort = l.sort,
    Date = dateadd(day, datediff(day, 0, p.DateTime), 0),
    Amount = p.Amount * g.coef
  from PaidOut p
    inner join @glcodes g on p.PaidOutCode = g.id
    inner join @locations l on p.Location = l.ID
  where p.DateTime >= @DateBegin
    and p.DateTime < @DateEnd + 1
) s
  inner join @dates d on s.Date = d.date
group by
  s.glcodesort,
  s.locationsort,
  d.sort;

insert into @glcodessales (glcodesort, locationsort, datesort, Amount)
select glcodesort, locationsort, 99999,    sum(Amount) from @glcodessales group by glcodesort, locationsort
union all
select glcodesort, 99999,        99999,    sum(Amount) from @glcodessales group by glcodesort
union all
select 999,        locationsort, datesort, sum(Amount) from @glcodessales group by locationsort, datesort
union all
select 999,        locationsort, 99999,    sum(Amount) from @glcodessales group by locationsort
union all
select 999,        99999,        99999,    sum(Amount) from @glcodessales group by case datesort when 0 then 0 end;

print '@servicemoney';
declare @servicemoney table (servicesort int, locationsort int, datesort int, Amount money);
insert into @servicemoney (servicesort, locationsort, datesort, Amount)
select
  s.sort,
  l.sort,
  d.sort,
  Amount = sum(ot.Price)
from [OIL STOP REPORT].[OIL STOP REPORT].dbo.OperativeTotals ot
  inner join [OIL STOP REPORT].[OIL STOP REPORT].dbo.OperativeTotalsItemsDescription otid
    on ot.Object = otid.Object and ot.ObjectID = otid.ObjectID
  inner join @services s on otid.HistoryCode = s.code
  inner join @dates d on ot.Date = d.date
  inner join @locations l on ot.Location = l.id
group by
  s.sort,
  l.sort,
  d.sort;

insert into @orders (Location, Date, Sale)
select l.ID, d.Date, -s.Amount
from @servicemoney s
  inner join @locations l on s.locationsort = l.sort
  inner join @dates d on s.datesort = d.sort;

insert into @servicemoney (servicesort, locationsort, datesort, Amount)
select servicesort, locationsort, 99999,    sum(Amount) from @servicemoney group by servicesort, locationsort
union all
select servicesort, 99999,        99999,    sum(Amount) from @servicemoney group by servicesort
union all
select 999,         locationsort, datesort, sum(Amount) from @servicemoney group by locationsort, datesort
union all
select 999,         locationsort, 99999,    sum(Amount) from @servicemoney group by locationsort
union all
select 999,         99999,        99999,    sum(Amount) from @servicemoney group by case datesort when 0 then 0 end;

print '@ordermoney';
declare @ordermoney table (locationsort int, datesort int, Sale money, Discount money, Tax money);
insert into @ordermoney (locationsort, datesort, Sale, Discount, Tax)
select
  l.sort,
  d.sort,
  Sale = sum(Sale),
  Discount = sum(Discount),
  Tax = sum(Tax)
from @orders o
  inner join @locations l on o.Location = l.ID
  inner join @dates d on o.Date = d.date
group by
  l.sort,
  d.sort;

insert into @ordermoney (locationsort, datesort, Sale, Discount, Tax)
select locationsort, 99999, sum(Sale), sum(Discount), sum(Tax) from @ordermoney group by locationsort
union all
select 99999,        99999, sum(Sale), sum(Discount), sum(Tax) from @ordermoney
union all
select 0,            99999, null,      null,          null;

if not exists (select * from @servicemoney)
  delete from @header where sort = 400;
if not exists (select * from @glcodessales)
  delete from @header where sort = 500;
if not exists (select * from @glcodesdiscounts)
  delete from @header where sort = 600;

if object_id('tempdb..#ReportSalesAnalysis') is not null drop table #ReportSalesAnalysis;

select
  ld.locationsort,
  ld.datesort,
  columnsort = h.sort + case h.sort
    when 400 then ss.sort
    when 500 then ggs.sort
    when 600 then ggd.sort
    else 0
  end,
  Amount =  
  case h.sort
    when 300 then isnull(o.Sale, 0)
    when 400 then isnull(s.Amount, 0)
    when 500 then isnull(gs.Amount, 0)
    when 600 then -isnull(gd.Amount, 0)
    when 700 then -isnull(o.Discount, 0)
    when 800 then isnull(o.Tax, 0)
    when 999 then isnull(o.Sale, 0) + isnull(s.Amount, 0) + isnull(gs.Amount, 0) - isnull(gd.Amount, 0) - isnull(o.Discount, 0) + isnull(o.Tax, 0)
  end,
  Caption =  
  case h.sort
    when 100 then case ll.sort when 0 then h.description else case dd.sort when 99999 then 'Total' else ll.code end end
    when 200 then case ll.sort when 0 then h.description else convert(varchar, dd.date, 101) end
    when 400 then ss.code
    when 500 then ggs.code + ' ' + ggs.name
    when 600 then ggd.code + ' (Discounts)'
    else h.description
  end
into #ReportSalesAnalysis
from (select distinct locationsort from @ordermoney) l
inner join (select distinct locationsort, datesort from @ordermoney) ld on l.locationsort = ld.locationsort
inner join @locations ll on l.locationsort = ll.sort
inner join @dates dd on ld.datesort = dd.sort
cross join @header h
left join @ordermoney o on h.sort in (300, 700, 800, 999)
  and ld.locationsort = o.locationsort
  and ld.datesort = o.datesort
left join (select distinct t.*, m.servicesort from @servicemoney m left join @services t on m.servicesort = t.sort) ss
  on h.sort = ss.servicesort or (h.sort = 400 and ss.servicesort < 100)
left join @servicemoney s on ss.servicesort = s.servicesort
  and ld.locationsort = s.locationsort  -- this line produces the error
-- the error is:
-- The multi-part identifier "ld.locationsort" could not be bound
  and ld.datesort = s.datesort
left join (select distinct t.*, m.glcodesort from @glcodessales m left join @glcodes t on m.glcodesort = t.sort) ggs
  on h.sort = ggs.glcodesort or (h.sort = 500 and ggs.glcodesort < 100)
left join @glcodessales gs on ggs.glcodesort = gs.glcodesort
  and ld.locationsort = gs.locationsort
  and ld.datesort = gs.datesort
left join (select distinct t.*, m.glcodesort from @glcodesdiscounts m left join @glcodes t on m.glcodesort = t.sort) ggd
  on h.sort = ggd.glcodesort or (h.sort = 600 and ggd.glcodesort < 100)
left join @glcodesdiscounts gd on ggd.glcodesort = gd.glcodesort
  and ld.locationsort = gd.locationsort
  and ld.datesort = gd.datesort;

select
  *, (select count(distinct columnsort) from #ReportSalesAnalysis) as ColumnCnt
from #ReportSalesAnalysis
order by
  ld.locationsort,
  ld.datesort,
  columnsort;
