using System; using System.Collections.Generic; using System.Text; using System.Drawing; using System.Text.RegularExpressions; using System.IO; using iTextSharp.text.pdf; using iTextSharp.text; using Itenso.Rtf; using Itenso.Rtf.Parser; using Itenso.Rtf.Interpreter; using Itenso.Rtf.Support; using Volian.Controls.Library; using VEPROMS.CSLA.Library; using Volian.Base.Library; namespace Volian.Print.Library { public partial class vlnParagraphs : List { private vlnParagraph _Parent; public vlnParagraph Parent { get { return _Parent; } set { _Parent = value; } } public vlnParagraphs(vlnParagraph parent) { _Parent = parent; } public float Add(PdfContentByte cb, ItemInfoList itemInfoList, float xoff, float yoff, float yoffRight, int rnoLevel, int maxRNO, FormatInfo formatInfo) { int? bxIndex = null; vlnBox box = null; float yTop = yoff; foreach (ItemInfo childItemInfo in itemInfoList) { // if this is a caution/note and it has a caution/note substep, do it before this caution/note, so that // it comes above it. if (childItemInfo.Cautions != null && (childItemInfo.IsCaution || childItemInfo.IsNote)) yoff = Add(cb, childItemInfo.Cautions, xoff, yoff, yoffRight, rnoLevel, maxRNO, formatInfo); if (childItemInfo.Notes != null && (childItemInfo.IsCaution || childItemInfo.IsNote)) yoff = Add(cb, childItemInfo.Notes, xoff, yoff, yoffRight, rnoLevel, maxRNO, formatInfo); int? bxIndx = childItemInfo.FormatStepData==null?-1:childItemInfo.FormatStepData.StepLayoutData.STBoxindex; if (bxIndx != -1 && (bxIndex != bxIndx || childItemInfo.FormatStepData.BoxIt)) { if (childItemInfo.FormatStepData.BoxIt) // this is a boxed HLS { box = new vlnBox(); box.MyBox = new Box(); box.DefBox = vlnBox.BoxThin; StepSectionLayoutData ssld = formatInfo.MyStepSectionLayoutData; int colR = int.Parse(formatInfo.MyStepSectionLayoutData.ColRTable.Split(",".ToCharArray())[childItemInfo.ColumnMode]); int widS = vlnPrintObject.ToInt(formatInfo.MyStepSectionLayoutData.WidSTablePrint, maxRNO); box.DefEnd = (float)(ssld.ColS + colR + widS + (60/4.8)); box.YOffset = yoff - .75F * vlnPrintObject.SixLinesPerInch; } else if (bxIndex == null) // First boxed step { box = new vlnBox(); box.MyBox = formatInfo.PlantFormat.FormatData.BoxList[(int)bxIndx]; if (box.MyBox != null) { box.YOffset = yoff; int ln = 1; // a format flag determines whether there is a space before the note/caution. if (childItemInfo.FormatStepData.OneLineBeforeTab) ln++; yoff += ln * vlnPrintObject.SixLinesPerInch; } } else // Change Box Style { if (box != null) box.Height = yoff - box.YOffset; // new height, with children box = new vlnBox(); box.MyBox = formatInfo.PlantFormat.FormatData.BoxList[(int)bxIndx]; box.YOffset = yoff; int ln = 1; // a format flag determines whether there is a space before the note/caution. if (childItemInfo.FormatStepData.OneLineBeforeTab) ln++; yoff += ln * vlnPrintObject.SixLinesPerInch; } bxIndex = bxIndx; } if (childItemInfo is SectionInfo) formatInfo = (childItemInfo as SectionInfo).LocalFormat ?? formatInfo; if (rnoLevel < maxRNO && childItemInfo.RNOs != null) yoff = Math.Max(yoff, yoffRight); vlnParagraph para = new vlnParagraph(Parent, cb, childItemInfo, xoff, yoff, rnoLevel, maxRNO, formatInfo); if (box != null && box.MyParent == null) { box.MyParent = para; para.PartsContainer.Add(box); } Add(para); // para.YBottomMost will have y for bottom of any substeps that are also enclosed in the box. if (childItemInfo.IsStep && childItemInfo.MyHLS.FormatStepData.UseSmartTemplate && para.ChildrenBelow.Count>0 && para.ChildrenBelow[0].YBottomMost > para.YBottomMost) yoff = para.ChildrenBelow[0].YBottomMost; else yoff = para.YBottomMost; if (childItemInfo.FormatStepData != null && childItemInfo.FormatStepData.BoxIt) { box.Height = yoff - box.YOffset - (1.1F * vlnPrintObject.SixLinesPerInch); box = null; // if doing boxed steps, only do single sibling at a time. } } if (box != null && box.MyBox != null) { box.Height = yoff - box.YOffset; // new height, with children yoff += 2 * vlnPrintObject.SixLinesPerInch; } return yoff; } public float ToPdf(PdfContentByte cb, float yPageStart, float yTopMargin, float yBottomMargin) { foreach (vlnParagraph child in this) { yPageStart = child.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); } return yPageStart; } } public partial class vlnParagraph : vlnPrintObject { private vlnTable _MyGrid; public vlnTable MyGrid { get { return _MyGrid; } set { _MyGrid = value; } } public float SectYPageStart = 0; private bool _IsWordDocPara = false; public bool IsWordDocPara { get { return _IsWordDocPara; } set { _IsWordDocPara = value; } } public float ParagraphToPdf(PdfContentByte cb, float yPageStart, float yTopMargin, float yBottomMargin) { if (Processed) return yPageStart; //float localYPageStart = yPageStart; Processed = true; if (_PartsAbove != null && _PartsAbove.Count > 0) yPageStart = PartsAbove.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); if (MyItemInfo.IsHigh && MyItemInfo.MyDocStyle.SpecialStepsFoldout) yPageStart -= SixLinesPerInch; float yLocation = CalculateYOffset(yPageStart, yTopMargin); if (MyItemInfo.HasChangeBar && MyPageHelper.ChangeBarDefinition.MyChangeBarType != PrintChangeBar.Without) MyPageHelper.AddChangeBar(DoChangeBar(cb, MyItemInfo, MyPageHelper, XOffset, yLocation, MyPageHelper.MaxRNO, MyItemInfo.ActiveFormat), cbMess); float retval = yLocation; if (MyItemInfo.IsFigure) { yLocation -= (SixLinesPerInch * MyPageHelper.YMultiplier); if (ImageText != null) retval = DrawFigure(cb, yBottomMargin, yLocation); else retval = DrawText(cb, ref yPageStart, yTopMargin, yBottomMargin, ref yLocation); } else if (!MyItemInfo.IsStepSection || (MyItemInfo.ActiveFormat.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ShowSectionTitles && !MyItemInfo.MyDocStyle.CancelSectTitle && !MyItemInfo.MyDocStyle.SpecialStepsFoldout)) // Don't ouput the Step Section title { bool doprint = true; // if this is a section, then check the section config for printable header too. if (MyItemInfo.IsSection) { SectionConfig sch = MyItemInfo.MyConfig as SectionConfig; if (!(MyItemInfo.ActiveFormat.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ShowSectionTitles && !MyItemInfo.MyDocStyle.CancelSectTitle && !MyItemInfo.MyDocStyle.SpecialStepsFoldout) && ((sch != null && sch.Section_PrintHdr != "Y") || !MyItemInfo.ActiveFormat.PlantFormat.FormatData.SectData.UseMetaSections)) doprint = false; } if (MyItemInfo.MyContent.MyGrid != null) retval = DrawGrid(cb, ref yPageStart, yTopMargin, yBottomMargin, ref yLocation); else if (doprint)retval = DrawText(cb, ref yPageStart, yTopMargin, yBottomMargin, ref yLocation); if (MyItemInfo.IsHigh) { MyPageHelper.PageBookmarks.Add(MyItemInfo, MyItemInfo.MyTab.CleanText + " " + MyItemInfo.DisplayText, new PdfDestination(PdfDestination.FITBH, yLocation + YVeryTop - YTopMost + SixLinesPerInch)); } } if (MyItemInfo.IsSection) { SectionConfig sc = MyItemInfo.MyConfig as SectionConfig; if (sc != null && sc.Section_TOC == "Y") { string tocKey = string.Format("TOC{0}", MyItemInfo.ItemID); if (MyPageHelper.MyTOCPageCounts.ContainsKey(tocKey)) { PageCount pc = MyPageHelper.MyTOCPageCounts[tocKey]; pc.Total = MyPageHelper.CurrentPageNumber; pc.DrawTemplates(); } } } //Height = yLocation - retval; if (_PartsLeft != null && _PartsLeft.Count > 0) yPageStart = PartsLeft.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); if (_PartsRight != null && _PartsRight.Count > 0) yPageStart = PartsRight.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); if (_PartsBelow != null && _PartsBelow.Count > 0) yPageStart = PartsBelow.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); if (_PartsContainer != null && _PartsContainer.Count > 0) yPageStart = PartsContainer.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); //if (localYPageStart != yPageStart) DebugText.WriteLine("ParToPdf-yPagestartDiff:{0},{1},{2}", MyItemInfo.ItemID, localYPageStart, yPageStart); return yPageStart; } private float DrawGrid(PdfContentByte cb, ref float yPageStart, float yTopMargin, float yBottomMargin, ref float yLocation) { //DebugText.WriteLine("{0},'{1}','{2}','<>'", MyItemInfo.ItemID, MyItemInfo.ShortPath, FormattedText); DebugText.WriteLine("{0},'{1}','{2}','<>',{3}", MyItemInfo.ItemID, MyItemInfo.ShortPath,MyItemInfo.MyContent.Text,XOffset); float retval = Rtf2Pdf.GridAt(cb, MyGrid, XOffset, yLocation, Width, 100, DebugInfo, yBottomMargin, !MyItemInfo.FormatStepData.Type.Contains("Borderless")); return retval; } private float DrawText(PdfContentByte cb, ref float yPageStart, float yTopMargin, float yBottomMargin, ref float yLocation) { DebugText.WriteLine("{0},'{1}','{2}','<>'", MyItemInfo.ItemID, MyItemInfo.DBSequence, FormattedText); //Console.WriteLine("{0},{1},'{2}','<>'", MyItemInfo.ItemID, MyItemInfo.DBSequence, IParagraph.Content); float retval = yLocation; // Check if only one line, i.e. "Height < (1.2F * IParagraph.Leading". The Leading can be for six or seven lines per inch, so the 1.2 // multiplier accounts for both. if (!MyItemInfo.IsStepSection && MyItemInfo.FormatStepData.CenterOneLineOnly && MyItemInfo.MyPrevious == null && MyItemInfo.NextItem == null && Height < (1.2F * IParagraph.Leading)) IParagraph.Alignment = Element.ALIGN_CENTER; if (!MyItemInfo.IsStepSection && MyItemInfo.FormatStepData.StepPrintData.Justify == "Center") IParagraph.Alignment = Element.ALIGN_CENTER; retval = Rtf2Pdf.TextAt(cb, IParagraph, XOffset, yLocation, Width, 100, DebugInfo, yBottomMargin); if (retval == 0) // problem occurred - paragraph was not able to be printed on page { // pagination logic needs to be fixed. ForcePagination(cb, ref yPageStart, yTopMargin, yBottomMargin, ref yLocation, ref retval); } return retval; } private static List myAttributes =new List(); private static void AddAttribute(string attr) { if (myAttributes.Contains(attr)) return; //Console.WriteLine("Attribute = \"{0}\"", attr); myAttributes.Add(attr); } private void CheckAttributes(System.Collections.Hashtable attributes) { foreach (string attr in attributes.Keys) { AddAttribute(attr); } } private string FormattedText { get { bool subscript = false; //if (_MyItemInfo.ItemID == 467) // Console.Write(""); StringBuilder sb = new StringBuilder(); //if (IParagraph.Chunks.Count > 1) // Console.WriteLine("{0} Chunks", IParagraph.Chunks.Count); foreach (Chunk chk in IParagraph.Chunks) { string prefix = ""; string suffix = ""; CheckAttributes(chk.Attributes); if (chk.Font.BaseFont != null && chk.Font.BaseFont.PostscriptFontName.ToUpper().Contains("BOLD")) { prefix += "\xD5"; suffix = "\xD6" + suffix; } if (chk.Attributes.ContainsKey("SUBSUPSCRIPT")) { float sup = (float)(chk.Attributes["SUBSUPSCRIPT"]); if (sup > 0) { prefix += "\xA6"; suffix = "\xD1" + suffix; } else if (sup < 0) { prefix += "\xD1"; suffix = "\xA6" + suffix; } } if (chk.Attributes.ContainsKey("UNDERLINE")) { prefix += "\x16"; suffix = "\x16" + suffix; ; } sb.Append(prefix + chk.Content + suffix); } string retval = sb.ToString(); retval = retval.Replace("\xD1\xA6", "");// Remove Multiple Superscript commands in a row retval = retval.Replace("\xA6\xD1", "");// Remove Multiple Superscript commands in a row retval = retval.Replace("\xD6\xD5", ""); retval = retval.Replace("\xD6\x16\xD5", "\x16"); return retval; } } //private bool _TextDebug = false; //true; this will turn on a writeline with debug in DrawText() private float DrawFigure(PdfContentByte cb, float yBottomMargin, float yLocation) { float retval = yLocation; if (ImageText != null) { string[] vals = ImageText.Split("\n".ToCharArray()); ProcedureInfo proc = MyItemInfo.MyProcedure; DocVersionInfo dvi = proc.ActiveParent as DocVersionInfo; ROFstInfo rofst = dvi.DocVersionAssociations[0].MyROFst; ROImageInfo roImage = ROImageInfo.GetByROFstID_FileName(rofst.ROFstID, vals[0]); if (roImage != null) { iTextSharp.text.Image it_image = iTextSharp.text.Image.GetInstance(roImage.Content); retval = Rtf2Pdf.FigureAt(cb, it_image, XOffset + MyItemInfo.FormatStepData.Font.CharsToTwips, yLocation, Width * MyPageHelper.YMultiplier, Height * MyPageHelper.YMultiplier, DebugInfo, yBottomMargin, !MyItemInfo.FormatStepData.Type.Contains("Borderless")); } } return retval; } private string DebugInfo { get { return string.Format("DebugID = {0}, ID={1} Type={2} TypeName='{3}' StepLevel={4} DBSequence={5}", DebugId, MyItemInfo.ItemID, MyItemInfo.FormatStepType, MyItemInfo.FormatStepData==null?"NoStepData":MyItemInfo.FormatStepData.Type, MyItemInfo.StepLevel, MyItemInfo.DBSequence); } } public override float ToPdf(PdfContentByte cb, float yPageStart, float yTopMargin, float yBottomMargin) { if (IsWordDocPara) { PdfReader tmp = null; string tmpstr = null; SectionInfo si = SectionInfo.Get(MyItemInfo.ItemID); cb.PdfDocument.NewPage(); MyPageHelper.MyPromsPrinter.CreateWordDocPdf(cb, si, ref tmp, ref tmpstr); Processed = true; return yPageStart; } float yLocalypagestart = yPageStart; // yPageStart is offset into roll; YTopMost is offset of topmost of this paragraph. float yLocation = yPageStart - YTopMost; DebugText.WriteLine("ToPdf1:{0},'{1}',{2},{3},{4},{5}", MyItemInfo.ItemID, MyItemInfo.ShortSearchPath, XOffset, yLocation, yPageStart, YTopMost); int paginate = Paginate(yLocation, yTopMargin, yBottomMargin); bool firstHighLevelStep = MyItemInfo.IsHigh && (MyItemInfo.MyPrevious == null); switch (paginate) { case 0: // No page break if (MyItemInfo.IsSection) { SectionInfo si = MyItemInfo as SectionInfo; MyPageHelper.PageBookmarks.Add(MyItemInfo, ((si.DisplayNumber ?? "") == "" ? "" : si.DisplayNumber + " - ") + si.DisplayText, null); DoCheckOffHeader(cb, MyItemInfo, yLocation, yTopMargin, yPageStart); } break; case 1: // Break on High Level Step OutputOtherPageSteps(cb, YTopMost, yPageStart, yTopMargin, yBottomMargin); cb.PdfDocument.NewPage(); DebugText.WriteLine("Paginate1"); if (MyItemInfo.IsSection) { SectionInfo si = MyItemInfo as SectionInfo; MyPageHelper.PageBookmarks.Add(MyItemInfo, ((si.DisplayNumber ?? "") == "" ? "" : si.DisplayNumber + " - ") + si.DisplayText, null); } // Only do foldout page if not done for section break, i.e. check the there's a previous step. if (MyItemInfo.MyPrevious != null && (MyItemInfo.ActiveSection.MyDocStyle.StructureStyle.Style ?? 0 & E_DocStructStyle.UseSectionFoldout) != 0) PromsPrinter.DoFoldoutPage(cb, "HLS", MyPageHelper.TextLayer, MyPageHelper); yPageStart = yTopMargin + YTopMost; DoCheckOffHeader(cb, MyItemInfo, yLocation, yTopMargin, yPageStart); MyPageHelper.YMultiplier = 1; break; case 2: // Break within a Step OutputOtherPageSteps(cb, YTopMost, yPageStart, yTopMargin, yBottomMargin); DocStyle docstyle = MyItemInfo.MyDocStyle; string myMsg = docstyle.Continue.Bottom.Message; if (myMsg != null && myMsg != "") { if (myMsg.IndexOf(@"%d") > -1) myMsg = myMsg.Replace(@"%d", MyItemInfo.MyHLS.MyTab.CleanText.Trim()); if (myMsg.IndexOf(@"%2d") > -1) myMsg = myMsg.Replace(@"%2d", MyItemInfo.MyHLS.MyTab.CleanText.Trim(" .".ToCharArray()).PadLeft(2)); float msg_yLocation = 0; if (myMsg.Contains("{par}")) { myMsg = myMsg.Replace("{par}", ""); msg_yLocation = -SixLinesPerInch; } switch (docstyle.Continue.Bottom.Location) { case E_ContBottomLoc.EndOfText: // place continue string at end of text case E_ContBottomLoc.BtwnTextAndBottom: // place continue string between end of text & bottom of page Console.WriteLine("Not done yet"); break; case E_ContBottomLoc.BottomOfPage: // place continue message at bottom of page //msg_yLocation = yBottomMargin + 2 * SixLinesPerInch + (float)docstyle.Layout.FooterLength; // 2 lines above bottom margin msg_yLocation = msg_yLocation + yBottomMargin + (float)docstyle.Layout.FooterLength; break; default: Console.WriteLine("**** BOTTOM CONTINUE MESSAGE NOT CODED*****"); break; } MyPageHelper.BottomMessage = new vlnText(cb, this, myMsg, myMsg, docstyle.Layout.LeftMargin + XOffsetBox + docstyle.Continue.Bottom.Margin ?? 0, msg_yLocation, docstyle.Continue.Bottom.Font);// MyItemInfo.ActiveFormat.PlantFormat.FormatData.Font); } cb.PdfDocument.NewPage(); DebugText.WriteLine("Paginate2"); if ((MyItemInfo.ActiveSection.MyDocStyle.StructureStyle.Style ?? 0 & E_DocStructStyle.UseSectionFoldout) != 0) PromsPrinter.DoFoldoutPage(cb, "Break within Step", MyPageHelper.TextLayer, MyPageHelper); // temporary foldout // If there is a box, adjust the yTopMost to include it. float yTopMost = YTopMost; //if (YVeryTop < yTopMost) Console.WriteLine("{0},{1},{2}", MyItemInfo.DBSequence, yTopMost, YVeryTop); yTopMost = Math.Min(yTopMost,YVeryTop); yPageStart = yTopMargin + yTopMost;// -2 * SixLinesPerInch; DoCheckOffHeader(cb, MyItemInfo, yLocation, yTopMargin, yPageStart); if (EmptyTopMostPart) yPageStart += SixLinesPerInch; myMsg = docstyle.Continue.Top.Message; if (myMsg != null && myMsg != "") { yPageStart -= 2 * SixLinesPerInch;// Allow two lines for top continue message if (myMsg.IndexOf(@"%d") > -1) myMsg = myMsg.Replace(@"%d", MyItemInfo.MyHLS.MyTab.CleanTextNoSymbols.Trim(" .".ToCharArray())); MyPageHelper.TopMessage = new vlnText(cb, this, myMsg, myMsg, docstyle.Layout.LeftMargin + XOffsetBox + docstyle.Continue.Top.Margin ?? 0, yTopMargin + 0.1F, docstyle.Continue.Top.Font);// MyItemInfo.ActiveFormat.PlantFormat.FormatData.Font); } MyPageHelper.YMultiplier = 1; break; case 3: // Break on High Level Step (SevenLinesPerInch) if (!firstHighLevelStep) { OutputOtherPageSteps(cb, YTopMost, yPageStart, yTopMargin, yBottomMargin); cb.PdfDocument.NewPage(); // HLS (7 lpi) breakif (MyItemInfo.IsSection) DebugText.WriteLine("Paginate3"); if (MyItemInfo.IsSection) { SectionInfo si = MyItemInfo as SectionInfo; MyPageHelper.PageBookmarks.Add(MyItemInfo, ((si.DisplayNumber ?? "") == "" ? "" : si.DisplayNumber + " - ") + si.DisplayText, null); } if ((MyItemInfo.ActiveSection.MyDocStyle.StructureStyle.Style ?? 0 & E_DocStructStyle.UseSectionFoldout) != 0) PromsPrinter.DoFoldoutPage(cb, "HLS (7 lpi) break", MyPageHelper.TextLayer, MyPageHelper); } if (MyItemInfo.MyParent != null && MyItemInfo.MyParent.IsStepSection && MyItemInfo.ActiveFormat.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ShowSectionTitles && !MyItemInfo.MyDocStyle.CancelSectTitle && !MyItemInfo.MyDocStyle.SpecialStepsFoldout) yPageStart = yPageStart; // if printing the section title, we already have the y location else yPageStart = yTopMargin + YTopMost; MyPageHelper.YMultiplier = _SevenLinesPerInch / SixLinesPerInch; DoCheckOffHeader(cb, MyItemInfo, yLocation, yTopMargin, yPageStart); break; } yPageStart = ChildrenAbove.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); yPageStart = ChildrenLeft.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); yPageStart = ParagraphToPdf(cb, yPageStart, yTopMargin, yBottomMargin); yPageStart = ChildrenRight.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); yPageStart = ChildrenBelow.ToPdf(cb, yPageStart, yTopMargin, yBottomMargin); if (MyItemInfo.IsHigh && MyItemInfo.NextItem == null) // last hls, add the 'end' message, if there is one { DocStyle docstyle = MyItemInfo.MyDocStyle; string myMsg = (docstyle.End == null) ? null : docstyle.End.Message; if (myMsg != null) { // If the flag is 0 or 1, just put the end message out right below this vlnParagraph: float msg_yLocation = CalculateYLocation(yPageStart - YBottomMost, yTopMargin); // use the 'flag' to position the message. if (docstyle.End.Flag > 2) // >2 position at an absolute location defined by docstyle.End.Flag. { msg_yLocation = yTopMargin - (float)(docstyle.End.Flag * SixLinesPerInch); } if (myMsg.Contains("{Section Number}")) myMsg = myMsg.Replace("{Section Number}", MyItemInfo.ActiveSection.DisplayNumber); // center the message. float wtpm = (float)docstyle.Layout.PageWidth - (float)docstyle.Layout.LeftMargin; float centerpos = XOffsetBox + (float)docstyle.Layout.LeftMargin + (wtpm - (myMsg.Length * MyItemInfo.FormatStepData.Font.CharsToTwips)) / 2; MyPageHelper.BottomMessage = new vlnText(cb, this, myMsg, myMsg, centerpos, msg_yLocation, docstyle.End.Font); } } if (yLocalypagestart != yPageStart) DebugText.WriteLine("ToPdf-yPagestartDiff:{0},{1},{2}", MyItemInfo.ItemID, yLocalypagestart, yPageStart); return yPageStart; } private vlnParagraph TopMostChild { get { if (ChildrenAbove.Count>0) return ChildrenAbove[0].TopMostChild; return this; } } private bool EmptyTopMostPart { get { if (TopMostChild.PartsAbove.Count == 0) return false; foreach (vlnPrintObject po in TopMostChild.PartsAbove) { vlnHeader hd = po as vlnHeader; if (hd == null) return false; if (hd.Text != "") return false; } return true; } } private void OutputOtherPageSteps(PdfContentByte cb, float YTopMost, float yPageStart, float yTopMargin, float yBottomMargin) { // Find any items remaining in MyPageHelper.MyParagraphs that should be printed on this page. vlnParagraphs process = new vlnParagraphs(null); foreach (vlnParagraph vPara in MyPageHelper.MyParagraphs.Values) { //if (!vPara.Processed && ((vPara.YOffset + vPara.Height) < YTopMost)) if (!vPara.Processed && ((vPara.YOffset) < YTopMost)) process.Add(vPara); } foreach (vlnParagraph vPara in process) { vPara.ParagraphToPdf(cb, yPageStart, yTopMargin, yBottomMargin); } } private float _YVeryTop = -1; public float YVeryTop { get { if (_YVeryTop == -1) { _YVeryTop = YTop; _YVeryTop = VeryTop(PartsAbove,_YVeryTop); _YVeryTop = VeryTop(PartsContainer,_YVeryTop); } return _YVeryTop; } } private float VeryTop(vlnPrintObjects parts, float yVeryTop) { if (parts != null) foreach (vlnPrintObject part in parts) if (part.YOffset < yVeryTop) yVeryTop = part.YOffset; return yVeryTop; } /// /// This gets the height of the step with it's Caution's, Notes and potentially any First Substeps /// /// private float GetFirstPieceSize() { vlnParagraph paraLast = GetFirstPieceLastPart(); float retval = (paraLast.YBottom) - YTopMost; //Console.WriteLine(MyItemInfo.DBSequence); return retval; } public override float YBottom { get { float bottom = YOffset + Height; if (PartsBelow != null) { foreach (vlnPrintObject part in PartsBelow) { float partBottom = part.YBottom; bottom = Math.Max(partBottom, bottom); } } return bottom; } } public float YBottomText { get { return YOffset + Height; } } private vlnParagraph GetFirstPieceLastPart() { vlnParagraph para = this; if (!MyItemInfo.ActiveFormat.MyStepSectionLayoutData.PaginateOnFirstSubstep && ChildrenBelow != null && ChildrenBelow.Count > 0) { para = ChildrenBelow[0].GetFirstPieceLastPart(); } if (ChildrenRight != null && ChildrenRight.Count > 0) { foreach(vlnParagraph paraRight in ChildrenRight) { vlnParagraph paraRightLast = paraRight.GetFirstPieceLastPart(); if (paraRightLast.YBottom > para.YBottom) para = paraRightLast; } } return para; } /// /// Find items from previous page (yLocation less than yTop) and remove them from the list. /// yTop is the new starting location on the next page for this step. /// /// /// private void RemoveProcessedParagraphs(StepLevelList myList, float yTop) { // the two CleanUpLists are used to remove items from // the original list because list items cannot be removed in a // 'foreach'. List CleanupListStepLevel = new List(); foreach(int stepLevel in myList.Keys) { List CleanupListYLocation = new List(); SortedList AdjustYLocation = new SortedList(); foreach (float yLocation in myList[stepLevel].Keys) { if (-yLocation <= yTop) CleanupListYLocation.Add(yLocation); else AdjustYLocation.Add(-yLocation, yLocation); // order in ascending order } foreach (float yLocation in CleanupListYLocation) myList[stepLevel].Remove(yLocation); foreach (float yLocation in AdjustYLocation.Keys) { vlnParagraph para = myList[stepLevel][-yLocation]; myList[stepLevel].Remove(-yLocation); // shift yLocation by yTop to work with items whose locations // are defined from the top of the new page rather than // from the beginning of the step. // Note that yLocation is negative to have items in descending // order so that adding yTop decrements their values. myList[stepLevel].Add(yTop-yLocation, para); } if (myList[stepLevel].Count == 0) CleanupListStepLevel.Add(stepLevel); } foreach (int stepLevel in CleanupListStepLevel) myList.Remove(stepLevel); } /// /// Builds a list of paragraphs by StepLevel and yLocation in descending order. /// /// /// private void BuildLocationList(float yTopMost, ParagraphLocations myLocations) { foreach (vlnParagraph child in ChildrenAbove) child.BuildLocationList(yTopMost, myLocations); foreach (vlnParagraph child in ChildrenLeft) child.BuildLocationList(yTopMost, myLocations); // Add a list entry consisting of StepLevel, yoffset from the beginning of this step, paragraph itself myLocations.Add(this, yTopMost); foreach (vlnParagraph child in ChildrenRight) child.BuildLocationList(yTopMost, myLocations); foreach (vlnParagraph child in ChildrenBelow) child.BuildLocationList(yTopMost, myLocations); } private int COL_WID_ADJ = 6; // adjusts for incorrect use of WidSTable when breaking a line (it breaks 6 chars too short) public vlnParagraph(vlnParagraph parent, PdfContentByte cb, ItemInfo itemInfo, float xoff, float yoff, int rnoLevel, int maxRNO, FormatInfo formatInfo) { float savCheckListBottomMost = 0; //int[] problemIDs = { 889 }; //List lProblemIDs = new List(problemIDs); //if (lProblemIDs.Contains(itemInfo.ItemID)) // Console.WriteLine("Found Item {0}", itemInfo.ItemID);formatInfochecko MyParent = parent; // The following code determines the last paragraph for an RNO // MyTopRNO finds the Top paragraph for an RNO if (itemInfo.IsInRNO) { if (rnoLevel <= maxRNO && itemInfo.IsRNOPart) // Not top level RNO MyTopRNO = this; else MyTopRNO = MyParent.MyTopRNO; if(MyTopRNO != null) MyTopRNO.LastRNO = this; } MyContentByte = cb; MyPageHelper.MyParagraphs.Add(itemInfo.ItemID, this); MyItemInfo = itemInfo; XOffset = xoff; if (!MyItemInfo.IsStep && !MyItemInfo.IsStepSection) { IsWordDocPara = true; // special case - this is a 'Word Doc' paragraph. The Processed flag is used to know // that steps need put out on a page for the pagination logic, and since there are no steps // we don't want to have to worry about putting this out for pagination logic. Processed = true; return; } YTopMost = YOffset = yoff; vlnTab mytab = null; bool doSectTab = false; // if this substep has a caution or note as its parent, then there may have been a special indent done on the // substep... check the CautionOrNoteSubstepIndent value, this is the additional indent amount in number // of characters. if (itemInfo.MyParent.IsCaution || itemInfo.MyParent.IsNote) XOffset += 72 * (itemInfo.FormatStepData.CautionOrNoteSubstepIndent == null ? 0 : (float)itemInfo.FormatStepData.CautionOrNoteSubstepIndent / (float)itemInfo.FormatStepData.Font.CPI); if (itemInfo.IsStep && itemInfo.MyHLS.FormatStepData.UseSmartTemplate && itemInfo.FormatStepData.StepLayoutData.AlignWithParentTab || (itemInfo.Steps != null && itemInfo.Steps.Count > 0 && itemInfo.Steps[0].FormatStepData.StepLayoutData.AlignWithParentTab)) itemInfo.MyTab = null; if (itemInfo.MyTab != null && itemInfo.MyTab.Text != null && itemInfo.MyTab.Text != "") { float localXOffset = XOffset; bool doprint = true; if (itemInfo.IsSection) { SectionConfig sch = MyItemInfo.MyConfig as SectionConfig; if (!(MyItemInfo.ActiveFormat.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ShowSectionTitles && !MyItemInfo.MyDocStyle.CancelSectTitle && !MyItemInfo.MyDocStyle.SpecialStepsFoldout) && ((sch != null && sch.Section_PrintHdr != "Y") || !MyItemInfo.ActiveFormat.PlantFormat.FormatData.SectData.UseMetaSections)) doprint = false; } if (doprint && itemInfo.IsSection && !itemInfo.MyDocStyle.CancelSectTitle && itemInfo.MyTab.Text.ToUpper() != "FOLDOUT") { doSectTab = true; if (itemInfo.IsStepSection && formatInfo.PlantFormat.FormatData.SectData.SectionNumber.Just == "PSLeft") { float offset = (float)itemInfo.MyDocStyle.Layout.LeftMargin; if (formatInfo.PlantFormat.FormatData.SectData.SectionNumber.Pos != null) { // determine level of metasections. This level will be used to calculate the // indent. DO I NEED THIS HERE???? int level = 0; ItemInfo iilvl = itemInfo; while (!iilvl.IsProcedure) { level++; iilvl = iilvl.MyParent; } level = level <= 2 ? 1 : level - 1; if (level == 1) offset += (float)formatInfo.PlantFormat.FormatData.SectData.SectionNumber.Pos; else offset += (level-1) * ((float)formatInfo.PlantFormat.FormatData.SectData.SectionHeader.Pos-(float)formatInfo.PlantFormat.FormatData.SectData.SectionNumber.Pos); } localXOffset = offset; } } if (!itemInfo.IsSection || doSectTab) { mytab = new vlnTab(cb, this, itemInfo.MyTab.Text, itemInfo.MyTab.CleanText, localXOffset, yoff, itemInfo.MyTab.MyFont, doSectTab, StepRTB.MySymbolFontName, itemInfo.MyTab.RemovedStyleUnderline); PartsLeft.Add(mytab); if (mytab.MyMacro != null) PartsLeft.Add(mytab.MyMacro); } } AdjustWidth(itemInfo, maxRNO, formatInfo, mytab); AdjustXOffsetForTab(itemInfo, maxRNO, formatInfo, mytab); if (itemInfo.MyHeader != null && itemInfo.MyHeader.Text != null && !doSectTab) yoff += SetHeader(this, cb, itemInfo, formatInfo); if (itemInfo.Cautions != null && !(itemInfo.IsCaution || itemInfo.IsNote)) yoff = ChildrenAbove.Add(cb, itemInfo.Cautions, xoff, yoff, yoff, rnoLevel, maxRNO, formatInfo); if (itemInfo.Notes != null && !(itemInfo.IsCaution || itemInfo.IsNote)) yoff = ChildrenAbove.Add(cb, itemInfo.Notes, xoff, yoff, yoff, rnoLevel, maxRNO, formatInfo); // If the format has that extra space should be put out before the step, then // handle this by using the 'PartsAbove' structure. By using the parts above, the extra // space above will be done regardless of where page breaks fall. float addExtraSpace = (MyItemInfo.FormatStepData == null) ? 0 : MyItemInfo.FormatStepData.StepLayoutData.STExtraSpace ?? 0; // If a high level step, the 16bit code uses the value of the extra space // from the high level step format regardless of what type of high level step it is: if (MyItemInfo.IsHigh) addExtraSpace = MyItemInfo.ActiveFormat.PlantFormat.FormatData.StepDataList[2].StepLayoutData.STExtraSpace ?? 0; if (addExtraSpace > 0) this.PartsAbove.Add(new vlnText(cb, this, " ", " ", 0, yoff, MyItemInfo.FormatStepData.Font)); yoff += addExtraSpace; YTop = yoff; YOffset = yoff; if (mytab != null && mytab.SeparateBullet) { string identB = formatInfo.PlantFormat.FormatData.SectData.StepSectionData.IdentB; vlnText myBullet = new vlnText(cb, this, identB, identB, XOffset - 12, YOffset, MyItemInfo.FormatStepData.TabData.Bullet.Font); myBullet.Rtf = myBullet.Rtf.Replace("\u25CF", @"\f1\u9679?\f0 "); PartsLeft.Add(myBullet); } if (itemInfo.IsRNOPart) { // there may be other places that double space, but this supports it for RNOs (the DoubleRNOspace flag) if (rnoLevel > maxRNO && itemInfo.FormatStepData.DoubleSpace) yoff = YOffset = yoff + SixLinesPerInch; } AddMacros(itemInfo, mytab); if (mytab != null) { mytab.YOffset = yoff; if (mytab.MyMacro != null) mytab.MyMacro.YOffset = yoff; } float yForCheckoff = yoff; //0; - default checkoff row is same as FIRST line of text // if dropCheckoff is true, then the checkoff is place on the same of row as the LAST line of text bool dropCheckoff = itemInfo.ActiveFormat.PlantFormat.FormatData.ProcData.CheckOffData.DropCheckOff; if (itemInfo.MyContent.MyGrid != null) { VlnFlexGrid myFlexGrid = new VlnFlexGrid(1,1); myFlexGrid.LoadGrid(itemInfo); MyGrid = new vlnTable(myFlexGrid, cb); Height = MyGrid.Height; Width = MyGrid.Width; //yForCheckoff = yoff + Height - SixLinesPerInch; if (dropCheckoff) yForCheckoff += Height - SixLinesPerInch; // place checkoff on last row of text yoff += (Height + (2 * SixLinesPerInch)); CalculateXOffset(itemInfo, maxRNO, formatInfo); } else if (itemInfo.IsFigure) // if a figure we've got to determine the size: { string erMsg = null; if (itemInfo.MyContent.Text != null) { ProcedureInfo proc = itemInfo.MyProcedure; // DocVersionInfo dvi = proc.ActiveParent as DocVersionInfo; DocVersionInfo dvi = proc.MyDocVersion; ROFstInfo rofst = dvi.DocVersionAssociations[0].MyROFst; rofst.docVer = dvi; ROFSTLookup lookup = rofst.ROFSTLookup; string linkInfoText = itemInfo.MyContent.Text.Replace(@"\v ", ""); if (linkInfoText.Contains("?")) erMsg = "RO was not found during data migration."; else { Match m = Regex.Match(linkInfoText, @"(.*)[#]Link:([A-Za-z]*):(.*)"); string[] subs = m.Groups[3].Value.Split(" ".ToCharArray()); string roid = subs[1]; string val = lookup.GetRoValue(subs[1]); if (val == null) val = lookup.GetRoValue(subs[1].Substring(0, 12)); if (val != null) { string[] vals = val.Split("\n".ToCharArray()); Width = Int32.Parse(vals[3], System.Globalization.NumberStyles.AllowHexSpecifier) * MyItemInfo.FormatStepData.Font.CharsToTwips; int lines = Int32.Parse(vals[2], System.Globalization.NumberStyles.AllowHexSpecifier); Height = lines * SixLinesPerInch; //yForCheckoff = yoff + Height - SixLinesPerInch; if (dropCheckoff) yForCheckoff += Height - SixLinesPerInch; // place checkoff on last row of text yoff += (Height + (2 * SixLinesPerInch)); try { ROImageInfo roImage = ROImageInfo.GetByROFstID_FileName(rofst.ROFstID, vals[0]); if (roImage != null) ImageText = val; else erMsg = string.Format("Image {0} does not exist.", vals[0]); } catch (Exception ex) { erMsg = string.Format("Image {0} does not exist, error = {1}.", vals[0], ex.Message); } } } if (erMsg != null) Rtf = GetRtf(erMsg, itemInfo.ActiveFormat.PlantFormat.FormatData.Font); CalculateXOffset(itemInfo, maxRNO, formatInfo); } } else { Rtf = GetRtf(itemInfo); // Need the following with some modifications for WCNCKL format: if (Rtf.Contains("{Backspace}")) { XOffset -= 22; Width += 24; Rtf = Rtf.Replace("{Backspace}", ""); } if (itemInfo.IsStep && itemInfo.MyHLS.FormatStepData.UseSmartTemplate) { if (itemInfo.FormatStepData.StepLayoutData.AlignWithParentTab || (itemInfo.Steps != null && itemInfo.Steps.Count > 0 && itemInfo.Steps[0].FormatStepData.StepLayoutData.AlignWithParentTab)) { int stplevl = TheStepLevel(itemInfo); if (stplevl >= 0 && MyItemInfo.MyHLS.FormatStepData.VertPos[stplevl] > 0) XOffset = float.Parse(MyItemInfo.MyHLS.FormatStepData.VertPos.Split(",".ToCharArray())[stplevl]) + (float)itemInfo.MyDocStyle.Layout.LeftMargin; } } if (itemInfo.IsTablePart) // Not for grid, this is for old-style tables. { Width = GetTableWidth(cb, IParagraph, MyItemInfo.MyDocStyle.Layout.PageWidth); CalculateXOffset(itemInfo, maxRNO, formatInfo); } else if (itemInfo.FormatStepData != null && itemInfo.FormatStepData.StepPrintData != null) XOffset += (float)(itemInfo.FormatStepData.StepPrintData.PosAdjust ?? 0); bool printMetaHdr = false; if (itemInfo.Steps != null && itemInfo.IsSection && itemInfo.Sections != null && itemInfo.Sections.Count > 0) { SectionConfig sc = itemInfo.MyConfig as SectionConfig; if ((sc != null && sc.Section_PrintHdr == "Y") && itemInfo.ActiveFormat.PlantFormat.FormatData.SectData.UseMetaSections) printMetaHdr = true; } // If WCN checklists, the substeps are printed in a row, need to keep track of the 'longest' in // y direction (bottommost) across the row. if (itemInfo.IsStep && itemInfo.MyHLS.FormatStepData.UseSmartTemplate && (TheStepLevel(itemInfo)>0)) { // do I need to get max of savCheckListBottomMost & (yoff+Height+SixLinesPerInch) savCheckListBottomMost = yoff + Height + SixLinesPerInch; } // Get Y offset for regular steps, or if section title is output or if not within row (not last column of // text) for wcn checklist, i.e. . if ((!itemInfo.IsStepSection && !itemInfo.MyHLS.FormatStepData.UseSmartTemplate) || printMetaHdr || (formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ShowSectionTitles && !MyItemInfo.MyDocStyle.CancelSectTitle && !MyItemInfo.MyDocStyle.SpecialStepsFoldout) // In WCN Checklist: I don't have children or if I have children the first child doesn't alignwithparent. || (!itemInfo.IsStepSection && itemInfo.MyHLS.FormatStepData.UseSmartTemplate && ((itemInfo.Steps == null || itemInfo.Steps.Count == 0) || !itemInfo.Steps[0].FormatStepData.StepLayoutData.AlignWithParentTab))) { yoff += Height; yoff += AdjustForBlankLines(); if (dropCheckoff) yForCheckoff += Height - SixLinesPerInch; // place checkoff on last row of text } } CheckOff co = itemInfo.GetCheckOffStep(); if (co != null) { //float xloc_co = (float)formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.XLocation + (float)itemInfo.MyDocStyle.Layout.LeftMargin; float xloc_co = (float)itemInfo.MyDocStyle.Layout.LeftMargin; if (formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.XLocation != null) xloc_co += (float)formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.XLocation; PartsRight.Add(new vlnMacro(xloc_co, yForCheckoff, co.Macro)); } float yOffRight = yoff; float RnoOffset = ToInt(formatInfo.MyStepSectionLayoutData.ColRTable, maxRNO); if (rnoLevel < maxRNO && itemInfo.RNOs != null) yOffRight = ChildrenRight.Add(cb, itemInfo.RNOs, XOffset + RnoOffset, YTop, YTop, rnoLevel + 1, maxRNO, formatInfo); // Need code to determine if the table will overlap the Right Column if it does then // use YOffRight rather than yoff if (itemInfo.Tables != null) { bool aerTableOrFigure = itemInfo.FormatStepData.Type.Contains("AER"); // get the first table & see if it is borderless. If borderless, 16bit code removed a // line (yoffset) before printing it: float yoffadj = 0; if (itemInfo.Tables[0].IsTable) { VlnFlexGrid fg = new VlnFlexGrid(1, 1); fg.LoadGrid(itemInfo.Tables[0]); if (fg.BorderStyle == C1.Win.C1FlexGrid.Util.BaseControls.BorderStyleEnum.None) yoffadj = -SixLinesPerInch; } if(!aerTableOrFigure && itemInfo.RNOLevel == 0) // Centered Table yoff = ChildrenBelow.Add(cb, itemInfo.Tables, XOffset, yOffRight, yOffRight, rnoLevel, maxRNO, formatInfo); else // AER or RNO Table yoff = ChildrenBelow.Add(cb, itemInfo.Tables, XOffset, yoff, yOffRight, rnoLevel, maxRNO, formatInfo); } // Look for the meta section case where the 'Editable' flag is set to 'N', which means that // these steps should not be printed: bool printsteps = true; if (itemInfo.Steps != null && itemInfo.IsSection && itemInfo.Sections!=null && itemInfo.Sections.Count>0) { SectionConfig sc = itemInfo.MyConfig as SectionConfig; if (sc != null && sc.SubSection_Edit != "Y") printsteps = false; } if (itemInfo.Steps != null && printsteps) yoff = ChildrenBelow.Add(cb, itemInfo.Steps, XOffset, yoff, yOffRight, rnoLevel, maxRNO, formatInfo); if (itemInfo.Sections != null) yoff = ChildrenBelow.Add(cb, itemInfo.Sections, xoff, yoff, yoff, rnoLevel, maxRNO, formatInfo); if (itemInfo.Procedures != null) yoff = ChildrenBelow.Add(cb, itemInfo.Procedures, xoff, yoff, yoff, rnoLevel, maxRNO, formatInfo); if (rnoLevel >= maxRNO && itemInfo.RNOs != null) yoff = ChildrenBelow.Add(cb, itemInfo.RNOs, XOffset, yoff, yoff, rnoLevel + 1, maxRNO, formatInfo); yoff = Math.Max(yoff, yOffRight); // TODO - use RNOSepAfterAER flag too: string tmpRnoSepStr = formatInfo.MyStepSectionPrintData.RNOSepString; if (rnoLevel < maxRNO && itemInfo.RNOs != null && tmpRnoSepStr != null) { vlnParagraph rno = ChildrenRight[0]; vlnParagraph bottomChild = BottomChild; float xsep = MyHighLevelParagraph.XOffset + RnoOffset; vlnRNOSeparator myRnoSep = new vlnRNOSeparator(this, cb, tmpRnoSepStr, xsep, yoff, formatInfo, bottomChild==null?true:bottomChild.MyItemInfo.HasChangeBar); // TODO: May need to handle more than one RNO column for RNO Separator if (rno.LastRNO.MyItemInfo.HasChangeBar == false && bottomChild != null && bottomChild.MyItemInfo.HasChangeBar) bottomChild.PartsBelow.Add(myRnoSep); else rno.LastRNO.PartsBelow.Add(myRnoSep); yoff += myRnoSep.Height + SixLinesPerInch; } YBottomMost = yoff; // For WCN Checklist, the substeps are in rows of data. The YBottomMost is the bottom most for // the row. if (savCheckListBottomMost != 0 && savCheckListBottomMost>YBottomMost) YBottomMost = savCheckListBottomMost; } private int TheStepLevel(ItemInfo itemInfo) { if (itemInfo.IsHigh) return -1; int lev = 0; ItemInfo ii = itemInfo; while (!ii.MyParent.IsHigh) { lev++; ii = ii.MyParent; } return lev; } private string SectionHasCheckOffHeader(ItemInfo itemInfo, ref VE_Font font) { // If the passed in itemInfo is not a section, get it's parent section. This may be a subsection, so // it will be found here, versus using 'MyActiveSection'. while (!itemInfo.IsSection) { itemInfo = itemInfo.MyParent; } // See if this section has a checkoffheading index - this value, if set is stored in the config field. // -1 flags no entries in the format's CheckOffHeaderList & 0 flags the first // entry which is always '{NO HEADING}'. So only continue if it's greater than 0. SectionInfo si = SectionInfo.Get(itemInfo.ItemID); // sometimes the itemInfo isn't a section info int sindx = si.CheckOffHeadingIndex(); // If there is a checkoff header for this section, see if this section has subsections. // If this has a header but no subsections, put out the header. // Else if this section has a header AND subsections, don't put it out... // the subsection's checkoff header will be processed (or lack there of). bool doCheckoffHeader = sindx > 0 && (itemInfo.Sections == null || itemInfo.Sections.Count == 0); // if doing it, then grab the string from the format.... CHECK if this works for subsection, it is // getting ActiveSection which may return the top section, not subsections. if (doCheckoffHeader && itemInfo.SectionCheckOffHeader != "") { font = itemInfo.ActiveFormat.PlantFormat.FormatData.ProcData.CheckOffData.CheckOffHeaderList[sindx].Font; return itemInfo.SectionCheckOffHeader; } return null; } private void DoCheckOffHeader(PdfContentByte cb, ItemInfo itemInfo, float yLocation, float yTopMargin, float yPageStart) { // CheckOffHeaders are handled in two different ways: // 1) Where the pagelist item puts them - this is at the 'top' of the page, when the current section on the // page has a checkoff header defined. Support for this is here AND in the VlnSvgPageHelper pagelist code. // 2) If a continuous section, and it doesn't start at 'top' of the page, then the checkoff header is // printed on the same line as the continuous section. Support for this is here, the check off header is added // to the 'PartRight' off of the section's VlnParagraph. // First see if there is any checkoff data in the format file and that there // is a pagelist item for the checkoff header. if (itemInfo.ActiveFormat.PlantFormat.FormatData.ProcData.CheckOffData.CheckOffList.Count <= 0) return; if (MyPageHelper.PageListCheckOffHeader == null) return; VE_Font font = null; ItemInfo mySectionInfo = itemInfo.MyActiveSection; ItemInfo mySubSectionInfo = itemInfo; // If the passed in itemInfo is not a section, get it's parent section. This may be a subsection, so // it will be found here, versus using 'MyActiveSection' which is the topmost section. The passed in // item may be a regular step, i.e. if a page break had occurred. if (!itemInfo.IsSection) { while (!mySubSectionInfo.IsSection) { mySubSectionInfo = mySubSectionInfo.MyParent; } } else mySubSectionInfo = (mySectionInfo.Sections == null)?null:mySectionInfo.Sections[0]; if (mySectionInfo == mySubSectionInfo) mySubSectionInfo = null; // The following duplicates the 16bit logic. The 16bit logic only put out a header if: // there is no subsection and section has a header (from config); // OR there is a subsection and the first subsection has a header. // For locating the header in 16bit logic: // If no subsections, output with section // If subsections and continuous section, put out on same line as subsection. (done here) // If subsections and starts at top of page, (on top of page, use pagelist code in VlnSvgPageHelper) // Get SectionInfos to access specific section config items... SectionInfo si = SectionInfo.Get(mySectionInfo.ItemID); int sindx = si.CheckOffHeadingIndex(); SectionInfo subi = mySubSectionInfo == null ? null : SectionInfo.Get(mySubSectionInfo.ItemID); int subindx = subi==null?-1:subi.CheckOffHeadingIndex(); // if there is no subsections & the main section doesn't use a header OR // if there is a subsection, & it does not use a header... Return. if ((subi==null&&sindx < 1) || (subi!=null&&subindx<1)) return; // figure out location now: // if step, it uses pagelist logic to locate it (this is a page break condition). // if continuous, put out with subsection if there is one, otherwise section. // otherwise, use pagelist logic. bool usePageListCOHdr = false; if (itemInfo.IsStep) usePageListCOHdr = true; else if (subi == null) //no subsection { if ((yPageStart - YOffset) == yTopMargin) usePageListCOHdr = true; } else { if (mySectionInfo == itemInfo && ((yPageStart - YOffset) == yTopMargin)) usePageListCOHdr = true; if (!usePageListCOHdr && mySubSectionInfo != itemInfo) return; } // grab the string from the format, depending on whether it should get it from the section or subsection. string cohead = (mySubSectionInfo!=null)?mySubSectionInfo.SectionCheckOffHeader:mySectionInfo.SectionCheckOffHeader; if (cohead == "" || cohead == null) return; // PageListCheckOffHeader is an svgtext - but really we only need the x/y location for this (pagelist support) // The actual text is from the checkoff header selection in the section config. if (cohead != MyPageHelper.PageListLastCheckOffHeader) { font = mySectionInfo.ActiveFormat.PlantFormat.FormatData.ProcData.CheckOffData.CheckOffHeaderList[sindx].Font; float chkOffY = CalculateYLocation(YTopMost, yTopMargin); if (MyPageHelper.YMultiplier != 1) { chkOffY = -1 + yTopMargin - (yTopMargin - yLocation) * MyPageHelper.YMultiplier; if (Rtf != null) IParagraph.Leading = _SevenLinesPerInch; } if (!usePageListCOHdr) { vlnText myCOHead = new vlnText(cb, this, cohead, cohead, MyPageHelper.PageListCheckOffHeader.X.Value + (float)MyItemInfo.MyDocStyle.Layout.LeftMargin, chkOffY, font); PartsRight.Add(myCOHead); } else MyPageHelper.PageListTopCheckOffHeader = cohead; MyPageHelper.PageListLastCheckOffHeader = cohead; } } internal vlnParagraph BottomChild { get { float bottomChildYBottomText = YBottomText; vlnParagraph bottomChild = this; foreach (vlnParagraph child in ChildrenRight) { vlnParagraph grandChild = child.BottomChild; if (grandChild != null && grandChild.YBottomText > bottomChildYBottomText) bottomChild = grandChild; if (grandChild != null && grandChild.YBottomText == bottomChildYBottomText && grandChild.MyItemInfo.HasChangeBar) bottomChild = grandChild; bottomChildYBottomText = bottomChild.YBottomText; } foreach (vlnParagraph child in ChildrenBelow) { vlnParagraph grandChild = child.BottomChild; if (grandChild != null && grandChild.YBottomText > bottomChildYBottomText) bottomChild = grandChild; if (grandChild != null && grandChild.YBottomText == bottomChildYBottomText && grandChild.MyItemInfo.HasChangeBar) bottomChild = grandChild; bottomChildYBottomText = bottomChild.YBottomText; } return bottomChild; } } private void CalculateXOffset(ItemInfo itemInfo, int maxRNO, FormatInfo formatInfo) { bool aerTableOrFigure = itemInfo.FormatStepData.Type.Contains("AER"); vlnParagraph hls1 = MyParent; while (hls1.MyParent != null && !hls1.MyItemInfo.IsHigh) hls1 = hls1.MyParent; float colR = float.Parse(formatInfo.MyStepSectionLayoutData.ColRTable.Split(",".ToCharArray())[itemInfo.ColumnMode]); float xLowerLimit = hls1.XOffset; float xUpperLimit = hls1.XOffset + hls1.Width + colR * itemInfo.ColumnMode; float TableCenterPos = float.Parse(formatInfo.MyStepSectionLayoutData.TableCenterPos.Split(",".ToCharArray())[itemInfo.ColumnMode]); if (formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.TieTabToLevel) { float pageWidth = ((float)itemInfo.MyActiveSection.MyDocStyle.Layout.PageWidth); //* (float)MyItemInfo.FormatStepData.Font.CPI / 12; float leftMargin = ((float)itemInfo.MyActiveSection.MyDocStyle.Layout.LeftMargin); //* (float)MyItemInfo.FormatStepData.Font.CPI / 12; XOffset = leftMargin + (pageWidth - leftMargin - Width) / 2; } else { // Determine center of hls XOffset = hls1.XOffset + hls1.Width / 2; // Determine left edge of the table by subtracting 1/2 of its width. XOffset -= Width / 2; if (aerTableOrFigure || itemInfo.RNOLevel != 0) // AER or RNO XOffset = MyParent.XOffset + MyParent.Width / 2 - Width / 2; else if (TableCenterPos != 0) XOffset = TableCenterPos; else // Centered Table or Figure { // Add in 1/2 of the width of all RNO columns XOffset += (colR * itemInfo.ColumnMode) / 2; if (!aerTableOrFigure) { // 05/14/12 - 16bit code adjusts the center depending upon the CPI of the table font. // Start with XOffset of table, then calculate the XOffset adjusted for the table font, // based on the default CPI of 12. Adjust the XOffset by the difference of these // two numbers. XOffset -= (XOffset - (XOffset * (float)MyItemInfo.FormatStepData.Font.CPI / 12)); } XOffset -= 12; // This makes the 16bit & 32bit align very closely. } // 05/14/12 - The PosAdjust is a format flag from 16bit. For WCN2, it appears that this amount equals // the amount of difference from a 10 CPI to a 12 CPI font. See comment above. if (itemInfo.FormatStepData != null && itemInfo.FormatStepData.StepPrintData != null) XOffset += (float)(itemInfo.FormatStepData.StepPrintData.PosAdjust ?? 0); } // if the XOffset < High Level Step Text's XOffset, then align with the High Level Step Text if (XOffset < xLowerLimit && Width < (xUpperLimit - xLowerLimit)) XOffset = xLowerLimit; // if the right margin exceeds the right edge of the rightmost RNO, then adjust right edge to match. if (XOffset + Width > xUpperLimit) XOffset = xUpperLimit - Width; // because of the above, if it pushes beyond the left margin, use the left margin. if (XOffset < (float)itemInfo.MyDocStyle.Layout.LeftMargin) XOffset = XOffsetBox = (float)itemInfo.MyDocStyle.Layout.LeftMargin; } private float AdjustForBlankLines() { if (MyItemInfo.MyDocStyle.SpecialStepsFoldout) return 0; int everyNLines = MyItemInfo.FormatStepData == null ? 1 : MyItemInfo.FormatStepData.StepLayoutData.EveryNLines ?? 1; if (MyItemInfo.Ordinal % everyNLines == 0 || MyItemInfo.NextItem == null) return SixLinesPerInch; return 0; } private void AddMacros(ItemInfo itemInfo, vlnTab mytab) { float y = YOffset; float x = mytab == null ? XOffset : mytab.XOffset + mytab.TabOffset; List myMacros = itemInfo.MyMacros; if (myMacros != null) { foreach (Macro myMacro in myMacros) { PartsLeft.Add(new vlnMacro(x, y, myMacro.MacroDef)); } } } // HLP Change bar flags in format file: // Locations: With Text, Outside Box, AER on Left RNO on Right, To The Left of Text // Text: Date & Change ID, Revision Number, Change ID, None, Custom (use input data) // Flag AERChgBarMsgRNOChgBarNoMsg and AERMsgRNONoMsg are NOT Used in 16-bit. private string cbMess = null; private vlnChangeBar DoChangeBar(PdfContentByte cb, ItemInfo itemInfo, VlnSvgPageHelper myPageHelper, float xoff, float yoff, int maxRNO, FormatInfo formatInfo) //, vlnChangeBar myCB) { // find column for the change bar based on format flags - this is code from 16-bit // if AbsoluteFixedChangeColumn // if FixedAERChangeColumn // if col>ColsS+ColR+COL_WID+ADJ // FixedChangeColumn // else // AERLeftChangeBarLocation() // else // FixedChangeColumn // else // ChangeBarLocation() ChangeBarData cbd = formatInfo.PlantFormat.FormatData.ProcData.ChangeBarData; float cols = formatInfo.MyStepSectionLayoutData.ColS ?? 0; float colr = ToInt(formatInfo.MyStepSectionLayoutData.ColRTable, maxRNO); // FixedChangeColumn + - Column location for change bars // 0 - Separate AER and RNO change bars to the right of the text // -10 to -1 - Change bars on left (specify # of columns from the text) // <-10 - AER change bars on the left and RNO change bars on the right. // FixedAERChangeColumn overrides the -5 default setting in chgbar.c, value is converted to a negative by the code float col = (cbd.AbsoluteFixedChangeColumn) ? ((cbd.FixedAERChangeColumn ?? 0) > 0) ? (xoff > (cols + colr + COL_WID_ADJ)) ? cbd.FixedChangeColumn ?? 0 : AERLeftChangeBarLocation(formatInfo) : cbd.FixedChangeColumn ?? 0 : ChangeBarLocation(xoff, this, formatInfo, maxRNO); if (myPageHelper.ChangeBarDefinition.MyChangeBarText == PrintChangeBarText.ChgID) cbMess = itemInfo.MyContent.UserID; else if (myPageHelper.ChangeBarDefinition.MyChangeBarText == PrintChangeBarText.DateChgID) { string fmtDate = itemInfo.MyContent.DTS.ToShortDateString(); if (fmtDate.Length != 10) // need to add zeros { if (fmtDate.IndexOf("/") == 1) fmtDate = "0" + fmtDate; if (fmtDate.IndexOf("/", 3) == 1) fmtDate = fmtDate.Substring(0, 3) + "0" + fmtDate.Substring(3, fmtDate.Length - 3); } cbMess = itemInfo.MyContent.UserID + @"\n" + fmtDate; } else if (myPageHelper.ChangeBarDefinition.MyChangeBarText == PrintChangeBarText.RevNum) cbMess = myPageHelper.Rev; else if (myPageHelper.ChangeBarDefinition.MyChangeBarText == PrintChangeBarText.UserDef) cbMess = myPageHelper.ChangeBarDefinition.MyChangeBarMessage; int msgAlign = Element.ALIGN_LEFT; if (myPageHelper.ChangeBarDefinition.MyChangeBarLoc == PrintChangeBarLoc.LeftOfText || (myPageHelper.ChangeBarDefinition.MyChangeBarLoc == PrintChangeBarLoc.AERleftRNOright && !itemInfo.IsInRNO)) msgAlign = Element.ALIGN_RIGHT; return new vlnChangeBar(cb, this, (float)itemInfo.MyDocStyle.Layout.LeftMargin + (col * itemInfo.FormatStepData.Font.CharsToTwips), yoff, msgAlign); } private int ChangeBarLocation(float c, vlnParagraph paragraph, FormatInfo formatInfo, int maxRNO) { int fixedChgCol = formatInfo.PlantFormat.FormatData.ProcData.ChangeBarData.FixedChangeColumn ?? 0; float cols = formatInfo.MyStepSectionLayoutData.ColS ?? 0; int colr = ToInt(formatInfo.MyStepSectionLayoutData.ColRTable, maxRNO); float tmpc = c; if (MyItemInfo.IsCaution || MyItemInfo.IsNote) { // Temporary fix for change bars on Notes and Cautions where the FixedChgCol < -10 tmpc = (float)formatInfo.MyStepSectionLayoutData.ColT + (float)formatInfo.MyStepSectionLayoutData.WidT; // end position of Caution / Note int typ = (int)(MyItemInfo.MyContent.Type % 10000); int? bxIndx = formatInfo.PlantFormat.FormatData.StepDataList[typ].StepLayoutData.STBoxindex; if (bxIndx != null) { Box bx = formatInfo.PlantFormat.FormatData.BoxList[(int)bxIndx]; if (bx != null) tmpc = (float)bx.Start + (float)bx.End; // end position of a boxed Caution / Note } } // if this is a caution or note, put the change bar to the right of the text: //if ((paragraph.MyItemInfo.IsCaution || paragraph.MyItemInfo.IsNote) && fixedChgCol < -10) return -fixedChgCol; if ((fixedChgCol == 0))// && paragraph.MyItemInfo.IsCaution || paragraph.MyItemInfo.IsNote) //if ((fixedChgCol==0) || paragraph.MyItemInfo.IsCaution || paragraph.MyItemInfo.IsNote) { float rightEdge = (paragraph.XOffset + paragraph.Width + 5); rightEdge-=(float)paragraph.MyItemInfo.MyDocStyle.Layout.LeftMargin; rightEdge = rightEdge/paragraph.MyItemInfo.FormatStepData.Font.CharsToTwips; return (int)rightEdge; } if (fixedChgCol < -10 || fixedChgCol >= 0) return ((fixedChgCol > 0) ? fixedChgCol : (fixedChgCol == 0) ? (int)c + 1 : (tmpc > (cols + colr + COL_WID_ADJ)) ? -fixedChgCol : //(c > (cols + colr + COL_WID_ADJ) || (MyItemInfo.IsCaution || MyItemInfo.IsNote)) ? -fixedChgCol : AERLeftChangeBarLocation(formatInfo)); else return (int)(fixedChgCol + (((c < cols + Width + colr) || TableTest() || MyItemInfo.IsCaution || MyItemInfo.IsNote) ? 0 : cols + colr)); // || (GetColumnMode() == 0)) ? 0 : cols + colr)); /* Change bars to left of text -- ColS+WidS+ColR is the end of RNO col*/ } /* ** Centered tables whose width exceeds the end of the RNO column should ** still have their change bars printed on the left. This procedure checks ** to make sure the step is a table and that it is not an RNO. */ private bool TableTest() { return false; //return ((s->Type == TABLE || s->Type == BORDERLESSTABLE) && //!(strchr((char *)&s->dbseq[2],RNO_MARKER))); } private float GetBottomYoff(float bottomY) { foreach (vlnParagraph child in ChildrenBelow) { if (child.YOffset > bottomY) bottomY = child.YOffset; bottomY = child.GetBottomYoff(bottomY); } return bottomY; } private int AERLeftChangeBarLocation(FormatInfo formatInfo) { // use -5 default unless it is format specified int fixedAERChgCol = formatInfo.PlantFormat.FormatData.ProcData.ChangeBarData.FixedAERChangeColumn ?? 0; return (fixedAERChgCol != 0) ? (fixedAERChgCol > 100) ? (fixedAERChgCol % 100) : -fixedAERChgCol : -5; } private StringBuilder _RtfSB = null; // return rtf and use for tab and other text public string GetRtf(ItemInfo itemInfo) { _RtfSB = new StringBuilder(); DisplayText vlntxt = new DisplayText(itemInfo, E_EditPrintMode.Print, E_ViewMode.View, true, E_FieldToEdit.StepText, false); System.Drawing.Font myFont = vlntxt.TextFont.WindowsFont; if (!itemInfo.IsTable && StepRTB.MyFontFamily != null) myFont = new System.Drawing.Font(StepRTB.MyFontFamily, myFont.Size, myFont.Style); if (itemInfo.IsHigh && itemInfo.MyDocStyle.UndSpecialStepsFoldout) myFont = new System.Drawing.Font(myFont.FontFamily, myFont.Size, myFont.Style | FontStyle.Underline); _RtfSB.Append(AddFontTable(myFont)); _RtfSB.Append(vlntxt.StartText); _RtfSB.Append("}"); return _RtfSB.ToString(); } private float _XOffsetBox = 0; public float XOffsetBox { get { return _XOffsetBox; } set { _XOffsetBox = value; } } private float SetHeader(vlnParagraph para, PdfContentByte cb, ItemInfo itemInfo, FormatInfo formatInfo) { float xoff = (float)itemInfo.MyDocStyle.Layout.LeftMargin; // The '6' below is really converting from characters to Twips. But instead of considering the CPI, 16bit assumed // it was 12 to center the header. That's where the 6 comes from. float hdrWidth = (itemInfo.MyHeader.CleanText == null) ? 0 : itemInfo.MyHeader.CleanText.Length * 6; int typ = ((int)itemInfo.MyContent.Type) % 10000; int? bxIndx = formatInfo.PlantFormat.FormatData.StepDataList[typ].StepLayoutData.STBoxindex; if (itemInfo.MyHeader.Justify == System.Drawing.ContentAlignment.MiddleCenter) { if (bxIndx != null) { Box bx = formatInfo.PlantFormat.FormatData.BoxList[(int)bxIndx]; if (bx.TabPos > 0) xoff += (float)bx.TabPos; // xoff starts as left margin else xoff += (float)((bx.TxtStart + XOffsetBox + (bx.TxtWidth / 2)) - (hdrWidth / 2)); // xoff starts as left margin } else if (formatInfo.MyStepSectionLayoutData.Separator.Location > 0) xoff = XOffset + AdjustToCharPosition((float)((para.Width - hdrWidth) / formatInfo.MyStepSectionLayoutData.Separator.Location), itemInfo.MyHeader.MyFont.CPI); else xoff = XOffset + (para.Width / 2) + (hdrWidth / 2); // XOffset has left margin included } else xoff = XOffset; // XOffset has left margin included vlnHeader myHeader = new vlnHeader(this, cb, itemInfo.MyHeader.Text, itemInfo.MyHeader.CleanText.TrimStart(" ".ToCharArray()), xoff, YOffset, itemInfo.MyHeader.MyFont); PartsAbove.Add(myHeader); return myHeader.Height + (!MyItemInfo.MyDocStyle.SpecialStepsFoldout ? SixLinesPerInch : 0); } private float AdjustToCharPosition(float position, float? CPI) { int iPosition = (int)position; int charsToTwips = (int)(72 / (CPI ?? 12)); iPosition = (iPosition / charsToTwips) * charsToTwips; position = (position / 7.2F) * 7.2F; return (float)iPosition; } private bool _SectionPageBreak = false; public bool SectionPageBreak { get { return _SectionPageBreak; } set { _SectionPageBreak = value; } } private bool _Processed = false; public bool Processed { get { return _Processed; } set { _Processed = value; } } private vlnParagraph _LastRNO; public vlnParagraph LastRNO { get { return _LastRNO; } set { _LastRNO = value; } } private vlnParagraph _MyTopRNO; public vlnParagraph MyTopRNO { get { return _MyTopRNO; } set { _MyTopRNO = value; } } private static string _Prefix = ""; public static string Prefix { get { return vlnParagraph._Prefix; } set { vlnParagraph._Prefix = value; } } protected float _YBottomMost; // Bottom of the paragraph including the children public float YBottomMost { get { return _YBottomMost; } set { _YBottomMost = value; } } protected float _YTopMost; // Top of the paragraph including the children public float YTopMost { get { return _YTopMost; } set { _YTopMost = value; } } protected float _YTop; // Top of the paragraph including parts public float YTop { get { return _YTop; } set { vlnParagraph vp = this as vlnParagraph; _YTop = value; } } public float YSize // How big this paragraph is with all of its children { get { return _YBottomMost - _YTopMost; } } private ItemInfo _MyItemInfo; public ItemInfo MyItemInfo { get { return _MyItemInfo; } set { _MyItemInfo = value; } } private vlnParagraph _MyHighLevelParagraph; public vlnParagraph MyHighLevelParagraph { get { if (_MyHighLevelParagraph == null) { _MyHighLevelParagraph = GetHighLevelParagraph(); } return _MyHighLevelParagraph; } } private vlnParagraph GetHighLevelParagraph() { if (MyItemInfo.IsHigh) return this; return MyParent.GetHighLevelParagraph(); } // Tab, Separator, ChangeBar, Box, Circle, Checkoff private vlnPrintObjects _PartsAbove; public vlnPrintObjects PartsAbove { get { if (_PartsAbove == null) _PartsAbove = new vlnPrintObjects(); return _PartsAbove; } } private vlnPrintObjects _PartsBelow; public vlnPrintObjects PartsBelow { get { if (_PartsBelow == null) _PartsBelow = new vlnPrintObjects(); return _PartsBelow; } } private vlnPrintObjects _PartsRight; public vlnPrintObjects PartsRight { get { if (_PartsRight == null) _PartsRight = new vlnPrintObjects(); return _PartsRight; } } private vlnPrintObjects _PartsLeft; public vlnPrintObjects PartsLeft { get { if (_PartsLeft == null) _PartsLeft = new vlnPrintObjects(); return _PartsLeft; } } private vlnPrintObjects _PartsContainer; public vlnPrintObjects PartsContainer { get { if (_PartsContainer == null) _PartsContainer = new vlnPrintObjects(); return _PartsContainer; } } private vlnParagraphs _ChildrenAbove; public vlnParagraphs ChildrenAbove { get { if (_ChildrenAbove == null) _ChildrenAbove = new vlnParagraphs(this); return _ChildrenAbove; } } private vlnParagraphs _ChildrenBelow; public vlnParagraphs ChildrenBelow { get { if (_ChildrenBelow == null) _ChildrenBelow = new vlnParagraphs(this); return _ChildrenBelow; } } private vlnParagraphs _ChildrenRight; public vlnParagraphs ChildrenRight { get { if (_ChildrenRight == null) _ChildrenRight = new vlnParagraphs(this); return _ChildrenRight; } } private vlnParagraphs _ChildrenLeft; public vlnParagraphs ChildrenLeft { get { if (_ChildrenLeft == null) _ChildrenLeft = new vlnParagraphs(this); return _ChildrenLeft; } } public void AdjustXOffsetForTab(ItemInfo itemInfo, int maxRNO, FormatInfo formatInfo, vlnTab myTab) { float tabWidth = (myTab == null) ? 0 : myTab.Width; if (itemInfo.IsStepSection) { if (formatInfo.PlantFormat.FormatData.SectData.SectionHeader.Just == "PSLeft") { int level = 0; ItemInfo iilvl = itemInfo; while (!iilvl.IsProcedure) { level++; iilvl = iilvl.MyParent; } level = level <= 2 ? 1 : level - 1; if (level==1) XOffset = (float)itemInfo.MyDocStyle.Layout.LeftMargin + (level * (float)formatInfo.PlantFormat.FormatData.SectData.SectionHeader.Pos); else XOffset = (float)formatInfo.PlantFormat.FormatData.SectData.MetaSectionList[level].ColSByLevel; } return; } int typ = ((int)itemInfo.MyContent.Type) % 10000; int? bxIndx = formatInfo.PlantFormat.FormatData.StepDataList[typ].StepLayoutData.STBoxindex; float? colOvrd = formatInfo.PlantFormat.FormatData.StepDataList[typ].ColOverride; if (bxIndx != null) { Box bx = formatInfo.PlantFormat.FormatData.BoxList[(int)bxIndx]; if (bx == null) { if ((colOvrd??0)!=0) { // 16-bit code subtracted the left margin //xoff = ((float)colOvrd - (float)itemInfo.MyDocStyle.Layout.LeftMargin) - XOffset; //XOffset += xoff; //if (myTab != null) myTab.XOffset += xoff; //xoff = MyParent.XOffset - myTab.XOffset; //XOffset += xoff; float tabOffset = (myTab == null?0: myTab.XOffset) - XOffset; XOffset = (float)colOvrd; if (myTab != null) myTab.XOffset = XOffset + tabOffset; return; } else XOffset = (float)itemInfo.MyDocStyle.Layout.LeftMargin + tabWidth + XOffsetBox; } else XOffset = (float)itemInfo.MyDocStyle.Layout.LeftMargin + (float)bx.TxtStart + tabWidth + XOffsetBox; if (myTab != null) myTab.XOffset = XOffset - tabWidth; } else if (itemInfo.IsHigh) { float x = 0; float xoff = 0; if ((colOvrd??0)!=0) x = (float)itemInfo.MyDocStyle.Layout.LeftMargin + (float)colOvrd; else x = (float)itemInfo.MyDocStyle.Layout.LeftMargin + (float)formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ColS; xoff = x - XOffset; // ColSByLevel will specify the column in which the High Level Step starts with // respect to the overall level calculation based on sections & meta-sections. // Only a few of the single column formats use this: WCN1, CAL1, RGEBCK if (formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.TieTabToLevel && !(itemInfo.MyDocStyle.SpecialStepsFoldout && itemInfo.MyDocStyle.UseColSByLevel)) { int indxLevel = itemInfo.PrintLevel + itemInfo.CurrentSectionLevel(); float colsbylevel = (float)formatInfo.PlantFormat.FormatData.SectData.MetaSectionList[indxLevel % formatInfo.PlantFormat.FormatData.SectData.MetaSectionList.Count].ColSByLevel; float seclvlindent = colsbylevel - (float)formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ColS; x = (float)itemInfo.MyDocStyle.Layout.LeftMargin + (float)formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ColS + seclvlindent; xoff = x - XOffset; if (myTab != null) myTab.XOffset = (float)itemInfo.MyDocStyle.Layout.LeftMargin + colsbylevel; LeftJustifyList jstlst = formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionPrintData.LeftJustifyList; if (jstlst.Count > 0) { float leftJustifyTabSize = jstlst[indxLevel % jstlst.Count].Size ?? 0; XOffset = myTab.XOffset + leftJustifyTabSize; } else XOffset = myTab.XOffset + (myTab.Text.Length * (float)itemInfo.FormatStepData.Font.CPI) - colsbylevel; } else { XOffset += xoff; if (myTab != null) { myTab.XOffset += xoff; if (myTab.MyMacro != null) myTab.MyMacro.XOffset += xoff; } } } else if (itemInfo.IsRNOPart && !((ItemInfo)itemInfo.ActiveParent).IsHigh) { // don't adjust for rno } else if (MyParent != null) { if (myTab != null && itemInfo.IsSequential && formatInfo.PlantFormat.FormatData.SectData.UseMetaSections && formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionPrintData.LeftJustifyList.Count > 0) { int indxLevels = itemInfo.PrintLevel + itemInfo.CurrentSectionLevel(); // kbr -1; LeftJustifyList jstlst = formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionPrintData.LeftJustifyList; float tableftadj = 0; foreach (LeftJustify lj in jstlst) { if (lj.Index == indxLevels) { tableftadj = lj.Size ?? 0; break; } } // the order of this is important. First adjust the tab's xoffset for the width of // the tab. The offset of text needs to be adjusted by the 'leftjustify' format variable // if it existed for this level. myTab.XOffset += tabWidth; if (tableftadj != 0 && myTab.Width0) widOvrd = xwid; else widOvrd = itemInfo.FormatStepData == null ? null : itemInfo.FormatStepData.WidthOverride; if ((widOvrd??0) != 0) { Width = (float)widOvrd; // if there's a box, we may need to do an adjustment on the width, if the tab data was changed // so don't return. if (bxIndx == null) return; } float tabWidth = (myTab == null) ? 0 : myTab.Width; if (bxIndx != null) { Box bx = formatInfo.PlantFormat.FormatData.BoxList[(int)bxIndx]; if (bx == null) { // NOT SURE IF WE NEED TO TEST FOR THE NULLBOX FORMAT FLAG, MAYBE "bx == null" IS THE SAME THING //if (formatInfo.MyStepSectionLayoutData.NullBox) Width = (float)formatInfo.MyStepSectionLayoutData.WidT - tabWidth; } else Width = _WidthAdjustBox + (float)bx.TxtWidth - tabWidth; // add 1 to get it to wrap like 16Bit } else if (itemInfo.IsHigh) { float CheckOffAdj = 0; if (itemInfo.MyDocStyle.UseCheckOffs) { if (formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.CheckOffAdjustment != null) CheckOffAdj = -(float)formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.CheckOffAdjustment; if (!HasCheckOffHeading(itemInfo, formatInfo) && CheckOffAdj < 0) CheckOffAdj += (float)(9 * 7.2); // 9 is the size of the SIGNOFF adjustment } float adjwidth = CheckOffAdj; SectData sd = formatInfo.PlantFormat.FormatData.SectData; if (sd.UseMetaSections) { if (sd.StepSectionData.StepSectionLayoutData.TieTabToLevel) { if (itemInfo.ActiveFormat.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ShowSectionTitles && !itemInfo.MyDocStyle.CancelSectTitle && !(itemInfo.MyDocStyle.SpecialStepsFoldout && itemInfo.MyDocStyle.UseColSByLevel)) { int indxLevel = itemInfo.PrintLevel + itemInfo.CurrentSectionLevel(); adjwidth += formatInfo.PlantFormat.FormatData.SectData.MetaSectionList[indxLevel % formatInfo.PlantFormat.FormatData.SectData.MetaSectionList.Count].WidSAdjByLevel ?? 0; ; float colsbylevel = (float)formatInfo.PlantFormat.FormatData.SectData.MetaSectionList[indxLevel % formatInfo.PlantFormat.FormatData.SectData.MetaSectionList.Count].ColSByLevel; float seclvlindent = colsbylevel - (float)formatInfo.PlantFormat.FormatData.SectData.StepSectionData.StepSectionLayoutData.ColS; adjwidth -= seclvlindent; adjwidth -= AdjustForSectionLevelTab(); } } else { if (formatInfo.PlantFormat.FormatData.SectData.MetaSectionList.Count > 0) adjwidth += (float)formatInfo.PlantFormat.FormatData.SectData.MetaSectionList[itemInfo.PrintLevel % formatInfo.PlantFormat.FormatData.SectData.MetaSectionList.Count].ColSByLevel; } } Width = ToInt(formatInfo.MyStepSectionLayoutData.WidSTablePrint, maxRNO); Width += _WidthAdjust; Width += adjwidth; } else if (itemInfo.IsCaution || itemInfo.IsNote) { float mycolT = (float)formatInfo.MyStepSectionLayoutData.ColT; Width = (float)formatInfo.MyStepSectionLayoutData.WidT - 6 - mycolT; XOffset += mycolT; // adjust caution/note text position if (PartsLeft != null)// adjust tab position foreach(vlnPrintObject vpo in PartsLeft) vpo.XOffset += mycolT; } else if (itemInfo.IsSection) { Width = _WidthAdjust + ToInt(formatInfo.MyStepSectionLayoutData.WidSTablePrint, maxRNO); } else if (MyParent == null) { // 72 points / inch - 7 inches (about width of page) Width = 72 * 7; } else if (itemInfo.IsRNOPart && !((ItemInfo)itemInfo.ActiveParent).IsHigh) { Width = MyParent.Width; } else if (itemInfo.IsTablePart) { Width = 72 * 7; // TODO: Need to determine the Width of the Table based upon the contents } else if (itemInfo.MyParent.IsCaution || itemInfo.MyParent.IsNote) { Width = MyParent.Width - tabWidth + (myTab == null ? 0 : myTab.TabAlign); Width -= 72 * (itemInfo.FormatStepData.CautionOrNoteSubstepIndent == null ? 0 : (float)itemInfo.FormatStepData.CautionOrNoteSubstepIndent / (float)itemInfo.FormatStepData.Font.CPI); } else { float adjwidth = 0; Width = adjwidth + MyParent.Width - tabWidth + (myTab == null ? 0 : myTab.TabAlign); } } private float GetWidthFromTemplate(ItemInfo itemInfo, FormatInfo formatInfo) { float rtnwid = 0; int tpIndx = GetSmartTemplateIndex(itemInfo, itemInfo.MyContent.Type, formatInfo); if (tpIndx != -1) { int ncol = formatInfo.PlantFormat.FormatData.Templates[tpIndx].width; // now convert to the units for this format. The template width data is in number of columns. rtnwid = ncol * itemInfo.FormatStepData.Font.CharsToTwips; } return rtnwid; } private int GetSmartTemplateIndex(ItemInfo itemInfo, int? type, FormatInfo formatInfo) { int indx = 0; // find the starting template in the list for the high level step type: int highType = (int)itemInfo.MyHLS.MyContent.Type-20001; while (indx < formatInfo.PlantFormat.FormatData.Templates.Count) { // level of 0 starts a new group. if (formatInfo.PlantFormat.FormatData.Templates[indx].level == 0) { indx++; // now see if we're on the high level step type, if so, look under this one // for the step type that we're on. If found return the index of it. if (formatInfo.PlantFormat.FormatData.Templates[indx].type == highType) { indx++; while (indx < formatInfo.PlantFormat.FormatData.Templates.Count && formatInfo.PlantFormat.FormatData.Templates[indx].level != 0) { if (formatInfo.PlantFormat.FormatData.Templates[indx].type == type - 20001) return indx; indx++; } } } else indx++; } return -1; // didn't find this step type in the template width override list. } private float AdjustForSectionLevelTab() { string sectTab = MyItemInfo.ActiveSection.MyTab.CleanText.TrimEnd(); if (sectTab.EndsWith(".0")) return 0; string myTab = MyItemInfo.MyTab.CleanText; // -1 below code, +1 for '.' and -2 for standard wid of tab // standard wid of this tab is 2 (ex: '1.') // so adjust width if longer than normal return (float)(sectTab.Length - 1) * 72/(float)MyItemInfo.FormatStepData.Font.CPI; } private bool HasCheckOffHeading(ItemInfo itemInfo, FormatInfo formatInfo) { if (formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.CheckOffHeaderList.Count != 0) { // does this section have a check off heading? SectionConfig sc = itemInfo.MyActiveSection.MyConfig as SectionConfig; if (sc != null && sc.Section_CheckoffHeaderSelection > 0) return true; } return false; } private bool UseCheckOffsDocStyle(int oldToNew, FormatInfo formatInfo) { int bit, retval=0; int mask=1; for(bit=0; bit < 32; bit++){ if ((oldToNew & mask) != 0 && ((int)formatInfo.PlantFormat.FormatData.ProcData.CheckOffData.UseCheckOffsIn & mask) !=0) retval = 1; mask <<= 1; } return retval > 0; } } /// /// First order by stepLevel, then within a stepLevel, order by descending yLocation on page. /// Want to find maximum y value that fits on page, i.e. put the most on page that can /// fit which is defined by the largest yLocation /// public class StepLevelList : SortedDictionary> { public StepLevelList() : base() { } public void Add(int stepLevel, float yLocation, vlnParagraph para) { if (stepLevel == 0) return; if (!this.ContainsKey(stepLevel)) this.Add(stepLevel, new SortedDictionary()); // using a negative for yLocation so that its in descending order: if (!this[stepLevel].ContainsKey(-yLocation)) this[stepLevel].Add(-yLocation, para); } } public class ParagraphLocations : List { public ParagraphLocations() : base() { } public void Add(vlnParagraph myParagraph, float yTopMost) { ParagraphLocation foundOverlap = FindOverlap(myParagraph); if (foundOverlap == null) { this.Add(new ParagraphLocation(yTopMost,myParagraph)); return; } ParagraphLocation doubleOverlap = FindOverlap(foundOverlap); while (doubleOverlap != null) { doubleOverlap.Merge(foundOverlap); this.Remove(foundOverlap); foundOverlap = doubleOverlap; doubleOverlap = FindOverlap(foundOverlap); } } private ParagraphLocation FindOverlap(vlnParagraph myParagraph) { foreach (ParagraphLocation paraLoc in this) if (paraLoc.Overlap(myParagraph)) return paraLoc; return null; } private ParagraphLocation FindOverlap(ParagraphLocation myParaLoc) { foreach (ParagraphLocation paraLoc in this) if (paraLoc != myParaLoc && paraLoc.Overlap(myParaLoc)) return paraLoc; return null; } internal StepLevelList BuildStepLevelList() { StepLevelList myList = new StepLevelList(); foreach (ParagraphLocation paraLoc in this) myList.Add(paraLoc.StepLevel, paraLoc.YTop, paraLoc.MyParagraph); return myList; } } public class ParagraphLocation { private float _YTop; public float YTop { get { return _YTop; } set { _YTop = value; } } private float _YBottom; public float YBottom { get { return _YBottom; } set { _YBottom = value; } } private int _StepLevel; public int StepLevel { get { return _StepLevel; } set { _StepLevel = value; } } private vlnParagraph _MyParagraph; public vlnParagraph MyParagraph { get { return _MyParagraph; } set { _MyParagraph = value; } } public ParagraphLocation(float yTopMost,vlnParagraph myParagraph) { MyParagraph =myParagraph; YTop = myParagraph.YVeryTop - yTopMost; YBottom = myParagraph.YBottom - yTopMost; StepLevel = myParagraph.MyItemInfo.StepLevel; } public bool Overlap(vlnParagraph otherParagraph) { if (Between(otherParagraph.YTop, YTop, YBottom)) return true; if (Between(otherParagraph.YBottom, YTop, YBottom)) return true; if(Between(YTop,otherParagraph.YVeryTop,otherParagraph.YBottom)) return true; if(Between(YBottom,otherParagraph.YVeryTop,otherParagraph.YBottom)) return true; return false; } public static bool Between(float x, float lower, float higher) { return x >= lower && x <= higher; } public bool Overlap(ParagraphLocation otherParagraphLocation) { if (Between(otherParagraphLocation.YTop, YTop, YBottom)) return true; // The top is within the other if (Between(otherParagraphLocation.YBottom, YTop, YBottom)) return true; // The bottom is within the other if(Between(YTop,otherParagraphLocation.YTop,otherParagraphLocation.YBottom)) return true; // the other top is within this one if(Between(YBottom,otherParagraphLocation.YTop,otherParagraphLocation.YBottom)) return true;// I believe this is unnecessary return false; } public void Merge(vlnParagraph otherParagraph) { if (StepLevel < otherParagraph.MyItemInfo.StepLevel) MyParagraph = otherParagraph; YTop = Math.Min(YTop, otherParagraph.YVeryTop); YBottom = Math.Max(YBottom, otherParagraph.YBottom); StepLevel = Math.Max(StepLevel, otherParagraph.MyItemInfo.StepLevel); } public void Merge(ParagraphLocation otherParagraphLocation) { if (StepLevel < otherParagraphLocation.StepLevel) MyParagraph = otherParagraphLocation.MyParagraph; YTop = Math.Min(YTop, otherParagraphLocation.YTop); YBottom = Math.Max(YBottom, otherParagraphLocation.YBottom); StepLevel = Math.Max(StepLevel, otherParagraphLocation.StepLevel); } } }