/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include <IDocumentContentOperations.hxx>
#include <IDocumentStylePoolAccess.hxx>
#include <charatr.hxx>
#include <editeng/postitem.hxx>
#include <editeng/crossedoutitem.hxx>
#include <editeng/udlnitem.hxx>
#include <editeng/wghtitem.hxx>
#include <editeng/fontitem.hxx>
#include <fmtanchr.hxx>
#include <fmtinfmt.hxx>
#include <i18nlangtag/languagetag.hxx>
#include <poolfmt.hxx>
#include <mdiexp.hxx>
#include <tools/urlobj.hxx>
#include "swmd.hxx"
int SwMarkdownParser::enter_block_callback(MD_BLOCKTYPE type, void* detail, void* userdata)
{
SwMarkdownParser* parser = static_cast<SwMarkdownParser*>(userdata);
::SetProgressState(parser->m_nProgress, parser->m_xDoc->GetDocShell());
switch (type)
{
case MD_BLOCK_DOC:
break;
case MD_BLOCK_QUOTE:
parser->AddBlockQuote();
break;
case MD_BLOCK_UL:
parser->StartNumberedBulletList(type);
break;
case MD_BLOCK_OL:
parser->StartNumberedBulletList(type);
break;
case MD_BLOCK_LI:
parser->StartNumberedBulletListItem(*static_cast<MD_BLOCK_LI_DETAIL*>(detail));
break;
case MD_BLOCK_HR:
parser->AddHR();
break;
case MD_BLOCK_H:
{
const MD_BLOCK_H_DETAIL* pDetail = static_cast<const MD_BLOCK_H_DETAIL*>(detail);
parser->StartHeading(pDetail->level);
break;
}
case MD_BLOCK_CODE:
parser->BeginCodeBlock();
break;
case MD_BLOCK_HTML:
parser->BeginHtmlBlock();
break;
case MD_BLOCK_P:
parser->StartPara();
break;
case MD_BLOCK_TABLE:
{
const MD_BLOCK_TABLE_DETAIL* pDetail
= static_cast<const MD_BLOCK_TABLE_DETAIL*>(detail);
const sal_Int32 nRow = pDetail->body_row_count + pDetail->head_row_count;
const sal_Int32 nCol = pDetail->col_count;
parser->StartTable(nRow, nCol);
break;
}
case MD_BLOCK_THEAD:
case MD_BLOCK_TBODY:
break;
case MD_BLOCK_TR:
parser->StartRow();
break;
case MD_BLOCK_TH:
case MD_BLOCK_TD:
const MD_BLOCK_TD_DETAIL* pDetail = static_cast<const MD_BLOCK_TD_DETAIL*>(detail);
parser->StartCell(pDetail->align);
break;
}
return 0;
}
int SwMarkdownParser::leave_block_callback(MD_BLOCKTYPE type, void* /*detail*/, void* userdata)
{
SwMarkdownParser* parser = static_cast<SwMarkdownParser*>(userdata);
switch (type)
{
case MD_BLOCK_DOC:
break;
case MD_BLOCK_QUOTE:
parser->EndBlockQuote();
break;
case MD_BLOCK_UL:
parser->EndNumberedBulletList();
break;
case MD_BLOCK_OL:
parser->EndNumberedBulletList();
break;
case MD_BLOCK_LI:
parser->EndNumberedBulletListItem();
break;
case MD_BLOCK_HR:
parser->EndHR();
break;
case MD_BLOCK_H:
parser->EndHeading();
break;
case MD_BLOCK_CODE:
parser->EndCodeBlock();
break;
case MD_BLOCK_HTML:
parser->EndHtmlBlock();
break;
case MD_BLOCK_P:
parser->EndPara();
break;
case MD_BLOCK_TABLE:
parser->EndTable();
break;
case MD_BLOCK_THEAD:
case MD_BLOCK_TBODY:
break;
case MD_BLOCK_TR:
case MD_BLOCK_TH:
break;
case MD_BLOCK_TD:
break;
}
return 0;
}
int SwMarkdownParser::enter_span_callback(MD_SPANTYPE type, void* detail, void* userdata)
{
SwMarkdownParser* parser = static_cast<SwMarkdownParser*>(userdata);
std::unique_ptr<SfxPoolItem> pItem;
switch (type)
{
case MD_SPAN_EM:
pItem.reset(new SvxPostureItem(ITALIC_NORMAL, RES_CHRATR_POSTURE));
break;
case MD_SPAN_STRONG:
pItem.reset(new SvxWeightItem(WEIGHT_BOLD, RES_CHRATR_WEIGHT));
break;
case MD_SPAN_A:
{
const MD_SPAN_A_DETAIL* pHyperlinkDetail = static_cast<const MD_SPAN_A_DETAIL*>(detail);
const MD_ATTRIBUTE& rHref = pHyperlinkDetail->href;
OUString aURL = rtl::OStringToOUString(std::string_view(rHref.text, rHref.size),
RTL_TEXTENCODING_UTF8);
aURL = INetURLObject::GetAbsURL(parser->m_sBaseURL, aURL);
pItem.reset(new SwFormatINetFormat(aURL, OUString()));
break;
}
case MD_SPAN_IMG:
{
const MD_SPAN_IMG_DETAIL* pImgDetail = static_cast<const MD_SPAN_IMG_DETAIL*>(detail);
const MD_ATTRIBUTE& rSrc = pImgDetail->src;
const MD_ATTRIBUTE& rTitle = pImgDetail->title;
OUString aURL = rtl::OStringToOUString(std::string_view(rSrc.text, rSrc.size),
RTL_TEXTENCODING_UTF8);
OUString aTitle = rtl::OStringToOUString(std::string_view(rTitle.text, rTitle.size),
RTL_TEXTENCODING_UTF8);
const SwFormatINetFormat* pINetFormat = nullptr;
if (!parser->m_aAttrStack.empty()
&& parser->m_aAttrStack.back()->Which() == RES_TXTATR_INETFMT)
{
// Copy the link to the image, so it won't be lost for empty spans.
const SfxPoolItem* pTopItem = parser->m_aAttrStack.back().get();
pINetFormat = dynamic_cast<const SwFormatINetFormat*>(pTopItem);
}
MDImage& rImg = parser->m_aImg;
rImg.url = std::move(aURL);
rImg.title = std::move(aTitle);
if (pINetFormat)
{
rImg.link = pINetFormat->GetValue();
}
parser->m_bInsideImage = true;
break;
}
case MD_SPAN_CODE:
{
SwDoc* pDoc = parser->m_xDoc.get();
SwTextFormatColl* pColl
= pDoc->getIDocumentStylePoolAccess().GetTextCollFromPool(RES_POOLCOLL_HTML_PRE);
pItem.reset(new SvxFontItem(pColl->GetFont()));
break;
}
case MD_SPAN_DEL:
pItem.reset(new SvxCrossedOutItem(STRIKEOUT_SINGLE, RES_CHRATR_CROSSEDOUT));
break;
case MD_SPAN_LATEXMATH:
case MD_SPAN_LATEXMATH_DISPLAY:
break;
case MD_SPAN_WIKILINK:
{
const MD_SPAN_WIKILINK_DETAIL* pWikilink
= static_cast<const MD_SPAN_WIKILINK_DETAIL*>(detail);
OUString target = rtl::OStringToOUString(
std::string_view(pWikilink->target.text, pWikilink->target.size),
RTL_TEXTENCODING_UTF8);
OUString aLang = GetAppLanguageTag().getLanguage();
OUString aURL = "https://" + aLang + ".wikipedia.org/wiki/" + target;
pItem.reset(new SwFormatINetFormat(aURL, OUString()));
break;
}
case MD_SPAN_U:
pItem.reset(new SvxUnderlineItem(LINESTYLE_SINGLE, RES_CHRATR_CROSSEDOUT));
break;
}
parser->m_aAttrStack.push_back(std::move(pItem));
return 0;
}
int SwMarkdownParser::leave_span_callback(MD_SPANTYPE type, void* /*detail*/, void* userdata)
{
SwMarkdownParser* parser = static_cast<SwMarkdownParser*>(userdata);
if (!parser->m_aAttrStack.empty())
parser->m_aAttrStack.pop_back();
switch (type)
{
case MD_SPAN_IMG:
parser->InsertCurImage();
parser->m_bInsideImage = false;
break;
case MD_SPAN_EM:
case MD_SPAN_STRONG:
case MD_SPAN_A:
case MD_SPAN_CODE:
case MD_SPAN_DEL:
case MD_SPAN_LATEXMATH:
case MD_SPAN_LATEXMATH_DISPLAY:
case MD_SPAN_WIKILINK:
case MD_SPAN_U:
break;
}
return 0;
}
int SwMarkdownParser::text_callback(MD_TEXTTYPE type, const MD_CHAR* text, MD_SIZE size,
void* userdata)
{
SwMarkdownParser* parser = static_cast<SwMarkdownParser*>(userdata);
parser->m_nProgress += size;
switch (type)
{
case MD_TEXT_HTML:
{
parser->m_htmlData += OStringToOUString({ text, size }, RTL_TEXTENCODING_UTF8);
break;
}
case MD_TEXT_CODE:
case MD_TEXT_NORMAL:
{
OUString aText = OStringToOUString({ text, size }, RTL_TEXTENCODING_UTF8);
if (!parser->m_bInsideImage)
parser->InsertText(aText);
else
parser->m_aImg.desc = std::move(aText);
break;
}
case MD_TEXT_BR:
parser->m_xDoc->getIDocumentContentOperations().InsertString(*parser->m_pPam,
u"\n"_ustr);
break;
case MD_TEXT_SOFTBR:
parser->m_xDoc->getIDocumentContentOperations().InsertString(*parser->m_pPam,
OUString(CHAR_HARDBLANK));
break;
case MD_TEXT_ENTITY:
case MD_TEXT_NULLCHAR:
case MD_TEXT_LATEXMATH:
break;
}
return 0;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s cinkeys+=0=break: */
↑ V1037 Two or more case-branches perform the same actions. Check lines: 51, 54
↑ V1037 Two or more case-branches perform the same actions. Check lines: 114, 117