| /* |
| * Copyright (C) 2014 Frédéric Wang (fred.wang@free.fr). All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "config.h" |
| |
| #if ENABLE(MATHML) |
| #include "RenderMathMLRadicalOperator.h" |
| |
| namespace WebCore { |
| |
| using namespace MathMLNames; |
| |
| static const UChar gRadicalCharacter = 0x221A; |
| |
| // This class relies on the RenderMathMLOperator class to draw a radical symbol. |
| // This does not work well when an OpenType MATH font is not available. |
| // In that case, we fallback to the old implementation of RenderMathMLRoot.cpp with graphic primitives. |
| |
| // Normal width of the front of the radical sign, before the base & overbar (em) |
| const float gFrontWidthEms = 0.75f; |
| // Horizontal position of the bottom point of the radical (* frontWidth) |
| const float gRadicalBottomPointXFront = 0.5f; |
| // Lower the radical sign's bottom point (px) |
| const int gRadicalBottomPointLower = 3; |
| // Horizontal position of the top left point of the radical "dip" (* frontWidth) |
| const float gRadicalDipLeftPointXFront = 0.8f; |
| // Vertical position of the top left point of a sqrt radical "dip" (* baseHeight) |
| const float gSqrtRadicalDipLeftPointYPos = 0.5f; |
| // Vertical shift of the left end point of the radical (em) |
| const float gRadicalLeftEndYShiftEms = 0.05f; |
| |
| // Radical line thickness (em) |
| const float gRadicalLineThicknessEms = 0.02f; |
| // Radical thick line thickness (em) |
| const float gRadicalThickLineThicknessEms = 0.1f; |
| |
| RenderMathMLRadicalOperator::RenderMathMLRadicalOperator(Document& document, Ref<RenderStyle>&& style) |
| : RenderMathMLOperator(document, WTF::move(style), String(&gRadicalCharacter, 1), MathMLOperatorDictionary::Prefix) |
| { |
| } |
| |
| void RenderMathMLRadicalOperator::stretchTo(LayoutUnit heightAboveBaseline, LayoutUnit depthBelowBaseline) |
| { |
| if (!style().fontCascade().primaryFont().mathData()) { |
| // If we do not have an OpenType MATH font, we always make the radical depth a bit larger than the target. |
| depthBelowBaseline += gRadicalBottomPointLower; |
| } |
| |
| RenderMathMLOperator::stretchTo(heightAboveBaseline, depthBelowBaseline); |
| } |
| |
| void RenderMathMLRadicalOperator::setOperatorProperties() |
| { |
| RenderMathMLOperator::setOperatorProperties(); |
| // We remove spacing around the radical symbol. |
| setLeadingSpace(0); |
| setTrailingSpace(0); |
| } |
| |
| void RenderMathMLRadicalOperator::computePreferredLogicalWidths() |
| { |
| ASSERT(preferredLogicalWidthsDirty()); |
| |
| if (style().fontCascade().primaryFont().mathData()) { |
| RenderMathMLOperator::computePreferredLogicalWidths(); |
| return; |
| } |
| |
| // If we do not have an OpenType MATH font, the front width is just given by the gFrontWidthEms constant. |
| int frontWidth = lroundf(gFrontWidthEms * style().fontSize()); |
| m_minPreferredLogicalWidth = frontWidth; |
| m_maxPreferredLogicalWidth = frontWidth; |
| } |
| |
| void RenderMathMLRadicalOperator::computeLogicalHeight(LayoutUnit logicalHeight, LayoutUnit logicalTop, LogicalExtentComputedValues& computedValues) const |
| { |
| if (style().fontCascade().primaryFont().mathData()) { |
| RenderMathMLOperator::computeLogicalHeight(logicalHeight, logicalTop, computedValues); |
| return; |
| } |
| |
| // If we do not have an OpenType MATH font, the logical height is always the stretch size. |
| logicalHeight = stretchSize(); |
| RenderBox::computeLogicalHeight(logicalHeight, logicalTop, computedValues); |
| } |
| |
| void RenderMathMLRadicalOperator::paint(PaintInfo& info, const LayoutPoint& paintOffset) |
| { |
| if (info.context().paintingDisabled() || info.phase != PaintPhaseForeground || style().visibility() != VISIBLE) |
| return; |
| |
| if (style().fontCascade().primaryFont().mathData()) { |
| RenderMathMLOperator::paint(info, paintOffset); |
| return; |
| } |
| |
| // If we do not have an OpenType MATH font, we paint the radical sign with graphic primitives. |
| IntPoint adjustedPaintOffset = roundedIntPoint(paintOffset + location() + contentBoxRect().location()); |
| int frontWidth = lroundf(gFrontWidthEms * style().fontSize()); |
| int startX = adjustedPaintOffset.x() + frontWidth; |
| int baseHeight = stretchSize() - gRadicalBottomPointLower; |
| |
| float radicalDipLeftPointYPos = gSqrtRadicalDipLeftPointYPos * baseHeight; |
| |
| FloatPoint overbarLeftPoint(startX, adjustedPaintOffset.y()); |
| FloatPoint bottomPoint(startX - gRadicalBottomPointXFront * frontWidth, adjustedPaintOffset.y() + baseHeight + gRadicalBottomPointLower); |
| FloatPoint dipLeftPoint(startX - gRadicalDipLeftPointXFront * frontWidth, adjustedPaintOffset.y() + radicalDipLeftPointYPos); |
| FloatPoint leftEnd(startX - frontWidth, dipLeftPoint.y() + gRadicalLeftEndYShiftEms * style().fontSize()); |
| |
| GraphicsContextStateSaver stateSaver(info.context()); |
| |
| info.context().setStrokeThickness(gRadicalLineThicknessEms * style().fontSize()); |
| info.context().setStrokeStyle(SolidStroke); |
| info.context().setStrokeColor(style().visitedDependentColor(CSSPropertyColor)); |
| info.context().setLineJoin(MiterJoin); |
| info.context().setMiterLimit(style().fontSize()); |
| |
| Path root; |
| |
| root.moveTo(FloatPoint(overbarLeftPoint.x(), adjustedPaintOffset.y())); |
| // draw from top left corner to bottom point of radical |
| root.addLineTo(bottomPoint); |
| // draw from bottom point to top of left part of radical base "dip" |
| root.addLineTo(dipLeftPoint); |
| // draw to end |
| root.addLineTo(leftEnd); |
| |
| info.context().strokePath(root); |
| |
| GraphicsContextStateSaver maskStateSaver(info.context()); |
| |
| // Build a mask to draw the thick part of the root. |
| Path maskPath; |
| |
| maskPath.moveTo(overbarLeftPoint); |
| maskPath.addLineTo(bottomPoint); |
| maskPath.addLineTo(dipLeftPoint); |
| maskPath.addLineTo(FloatPoint(2 * dipLeftPoint.x() - leftEnd.x(), 2 * dipLeftPoint.y() - leftEnd.y())); |
| |
| info.context().clipPath(maskPath); |
| |
| // Draw the thick part of the root. |
| info.context().setStrokeThickness(gRadicalThickLineThicknessEms * style().fontSize()); |
| info.context().setLineCap(SquareCap); |
| |
| Path line; |
| line.moveTo(bottomPoint); |
| line.addLineTo(dipLeftPoint); |
| |
| info.context().strokePath(line); |
| } |
| |
| } |
| |
| #endif |