主要是当我们缩短/截断文本内容时,我们通常只是在特定字符索引处截断它。无论如何,HTML已经很复杂,但我想使用不同的措施截断我的HTML内容(使用内容可编辑的div
生成):
N
,它将用作截断起点限制 N
个字符(仅文本;不计算标签);如果不是,它将只返回整个内容N-X
到N+X
字符位置(仅文本)并搜索块节点的末尾; X
是预定义的偏移值,可能约为N/5
到N/4
; N
N
的索引并截断该位置。我的内容可编辑生成内容可能包含段落(带换行符),预先格式化的代码块,块引号,有序和无序列表,标题,粗体和斜体(这些是内联节点,在截断时不应计算)最终的实现当然会定义哪些元素具体是可能的截断候选者。标题即使它们是块HTML元素也不会算作截断点,因为我们不想要寡居标题。段落,列出单个项目,整个有序和无序列表,块引号,预格式化块,无效元素等都是好的。标题和所有内联块元素都不是。
让我们把这个stackoverflow问题作为我想要截断的HTML内容的一个例子。我们将截断限制设置为 1000 ,偏移量为 250 个字符(1/4)。
This DotNetFiddle显示此问题的文字,同时在其中添加限制标记(|MIN|
代表字符750,|LIMIT|
代表字符1000,|MAX|
代表字符1250 )。
从示例可以看出,两个块节点与字符1000之间最近的截断边界在</OL>
和P
之间(我的内容可编辑生成。 .. 的)。这意味着我的HTML应该在这两个标记之间被截断,这将导致内容文本明显少于1000个字符,但保留截断的内容是有意义的,因为它不会截断某些文本中间的某个地方通道。
我希望这能解释与此算法相关的工作方式。
我在这里看到的第一个问题是我正在处理像HTML这样的嵌套结构。我还必须检测不同的元素(只有块元素,没有内联元素)。最后但并非最不重要的是,我必须只计算字符串中的某些字符并忽略那些属于标记的字符。
N
的新行并转换回HTML 如何处理这种截断算法?我的头脑似乎太累了,无法达成共识(或解决方案)。
答案 0 :(得分:3)
以下是一些可以截断内部文本的示例代码。它使用InnerText
属性和CloneNode
方法的递归功能。
public static HtmlNode TruncateInnerText(HtmlNode node, int length)
{
if (node == null)
throw new ArgumentNullException("node");
// nothing to do?
if (node.InnerText.Length < length)
return node;
HtmlNode clone = node.CloneNode(false);
TruncateInnerText(node, clone, clone, length);
return clone;
}
private static void TruncateInnerText(HtmlNode source, HtmlNode root, HtmlNode current, int length)
{
HtmlNode childClone;
foreach (HtmlNode child in source.ChildNodes)
{
// is expected size is ok?
int expectedSize = child.InnerText.Length + root.InnerText.Length;
if (expectedSize <= length)
{
// yes, just clone the whole hierarchy
childClone = child.CloneNode(true);
current.ChildNodes.Add(childClone);
continue;
}
// is it a text node? then crop it
HtmlTextNode text = child as HtmlTextNode;
if (text != null)
{
int remove = expectedSize - length;
childClone = root.OwnerDocument.CreateTextNode(text.InnerText.Substring(0, text.InnerText.Length - remove));
current.ChildNodes.Add(childClone);
return;
}
// it's not a text node, shallow clone and dive in
childClone = child.CloneNode(false);
current.ChildNodes.Add(childClone);
TruncateInnerText(child, root, childClone, length);
}
}
一个示例C#控制台应用程序将废弃此问题作为示例,并将其截断为500个字符。
class Program
{
static void Main(string[] args)
{
var web = new HtmlWeb();
var doc = web.Load("http://stackoverflow.com/questions/30926684/truncating-html-content-at-the-end-of-text-blocks-block-elements");
var post = doc.DocumentNode.SelectSingleNode("//td[@class='postcell']//div[@class='post-text']");
var truncated = TruncateInnerText(post, 500);
Console.WriteLine(truncated.OuterHtml);
Console.WriteLine("Size: " + truncated.InnerText.Length);
}
}
运行时,它应显示:
<div class="post-text" itemprop="text">
<p>Mainly when we shorten/truncate textual content we usually just truncate it at specific character index. That's already complicated in HTML anyway, but I want to truncate my HTML content (generated using content-editable <code>div</code>) using different measures:</p>
<ol>
<li>I would define character index <code>N</code> that will serve as truncating startpoint <em>limit</em></li>
<li>Algorithm will check whether content is at least <code>N</code> characters long (text only; not counting tags); if it's not it will just return the whole content</li>
<li>It would then</li></ol></div>
Size: 500
注意:我没有在字边界处截断,只是在字符边界处,并且没有,根据我的评论中的建议,它根本没有: - )
答案 1 :(得分:0)
private void RemoveEmpty(HtmlNode node){
var parent = node.Parent;
node.Remove();
if(parent==null)
return;
// remove parent if it is empty
if(!parent.DescendantNodes.Any()){
RemoveEmpty(parent);
}
}
private void Truncate(DocumentNode root, int maxLimit){
var n = 0;
HtmlTextNode lastNode = null;
foreach(var node in root.DescendantNodes
.OfType<HtmlTextNode>().ToArray()){
var length = node.Text.Length;
n+= length;
if(n + length >= maxLimit){
RemoveEmpty(node);
}
}
}
// you are left with only nodes that add up to your max limit characters.
答案 2 :(得分:-1)
我将遍历整个DOM树并继续计算出现的文本字符数。每当我达到极限(N)时,我将删除该文本节点的额外字符,从那里我将删除所有文本节点。
我相信这是一种保持所有HTML + CSS结构同时只保留N个字符的安全方法。