epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 1 | |
| 2 | /* |
| 3 | * Copyright 2011 Google Inc. |
| 4 | * |
| 5 | * Use of this source code is governed by a BSD-style license that can be |
| 6 | * found in the LICENSE file. |
| 7 | */ |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 8 | #include "SkStackViewLayout.h" |
| 9 | |
| 10 | SkStackViewLayout::SkStackViewLayout() |
| 11 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 12 | fMargin.set(0, 0, 0, 0); |
| 13 | fSpacer = 0; |
| 14 | fOrient = kHorizontal_Orient; |
| 15 | fPack = kStart_Pack; |
| 16 | fAlign = kStart_Align; |
| 17 | fRound = false; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 18 | } |
| 19 | |
| 20 | void SkStackViewLayout::setOrient(Orient ori) |
| 21 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 22 | SkASSERT((unsigned)ori < kOrientCount); |
| 23 | fOrient = SkToU8(ori); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 24 | } |
| 25 | |
| 26 | void SkStackViewLayout::getMargin(SkRect* margin) const |
| 27 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 28 | if (margin) |
| 29 | *margin = fMargin; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 30 | } |
| 31 | |
| 32 | void SkStackViewLayout::setMargin(const SkRect& margin) |
| 33 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 34 | fMargin = margin; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 35 | } |
| 36 | |
| 37 | void SkStackViewLayout::setSpacer(SkScalar spacer) |
| 38 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 39 | fSpacer = spacer; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 40 | } |
| 41 | |
| 42 | void SkStackViewLayout::setPack(Pack pack) |
| 43 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 44 | SkASSERT((unsigned)pack < kPackCount); |
| 45 | fPack = SkToU8(pack); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 46 | } |
| 47 | |
| 48 | void SkStackViewLayout::setAlign(Align align) |
| 49 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 50 | SkASSERT((unsigned)align < kAlignCount); |
| 51 | fAlign = SkToU8(align); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 52 | } |
| 53 | |
| 54 | void SkStackViewLayout::setRound(bool r) |
| 55 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 56 | fRound = SkToU8(r); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 57 | } |
| 58 | |
| 59 | //////////////////////////////////////////////////////////////////////////////// |
| 60 | |
| 61 | typedef SkScalar (*AlignProc)(SkScalar childLimit, SkScalar parentLimit); |
| 62 | typedef SkScalar (SkView::*GetSizeProc)() const; |
| 63 | typedef void (SkView::*SetLocProc)(SkScalar coord); |
| 64 | typedef void (SkView::*SetSizeProc)(SkScalar coord); |
| 65 | |
| 66 | static SkScalar left_align_proc(SkScalar childLimit, SkScalar parentLimit) { return 0; } |
| 67 | static SkScalar center_align_proc(SkScalar childLimit, SkScalar parentLimit) { return SkScalarHalf(parentLimit - childLimit); } |
| 68 | static SkScalar right_align_proc(SkScalar childLimit, SkScalar parentLimit) { return parentLimit - childLimit; } |
| 69 | static SkScalar fill_align_proc(SkScalar childLimit, SkScalar parentLimit) { return 0; } |
| 70 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 71 | /* Measure the main-dimension for all the children. If a child is marked flex in that direction |
| 72 | ignore its current value but increment the counter for flexChildren |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 73 | */ |
| 74 | static SkScalar compute_children_limit(SkView* parent, GetSizeProc sizeProc, int* count, |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 75 | uint32_t flexMask, int* flexCount) |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 76 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 77 | SkView::B2FIter iter(parent); |
| 78 | SkView* child; |
| 79 | SkScalar limit = 0; |
| 80 | int n = 0, flex = 0; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 81 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 82 | while ((child = iter.next()) != NULL) |
| 83 | { |
| 84 | n += 1; |
| 85 | if (child->getFlags() & flexMask) |
| 86 | flex += 1; |
| 87 | else |
| 88 | limit += (child->*sizeProc)(); |
| 89 | } |
| 90 | if (count) |
| 91 | *count = n; |
| 92 | if (flexCount) |
| 93 | *flexCount = flex; |
| 94 | return limit; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 95 | } |
| 96 | |
| 97 | void SkStackViewLayout::onLayoutChildren(SkView* parent) |
| 98 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 99 | static AlignProc gAlignProcs[] = { |
| 100 | left_align_proc, |
| 101 | center_align_proc, |
| 102 | right_align_proc, |
| 103 | fill_align_proc |
| 104 | }; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 105 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 106 | SkScalar startM, endM, crossStartM, crossLimit; |
| 107 | GetSizeProc mainGetSizeP, crossGetSizeP; |
| 108 | SetLocProc mainLocP, crossLocP; |
| 109 | SetSizeProc mainSetSizeP, crossSetSizeP; |
| 110 | SkView::Flag_Mask flexMask; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 111 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 112 | if (fOrient == kHorizontal_Orient) |
| 113 | { |
| 114 | startM = fMargin.fLeft; |
| 115 | endM = fMargin.fRight; |
| 116 | crossStartM = fMargin.fTop; |
| 117 | crossLimit = -fMargin.fTop - fMargin.fBottom; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 118 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 119 | mainGetSizeP = &SkView::width; |
| 120 | crossGetSizeP = &SkView::height; |
| 121 | mainLocP = &SkView::setLocX; |
| 122 | crossLocP = &SkView::setLocY; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 123 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 124 | mainSetSizeP = &SkView::setWidth; |
| 125 | crossSetSizeP = &SkView::setHeight; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 126 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 127 | flexMask = SkView::kFlexH_Mask; |
| 128 | } |
| 129 | else |
| 130 | { |
| 131 | startM = fMargin.fTop; |
| 132 | endM = fMargin.fBottom; |
| 133 | crossStartM = fMargin.fLeft; |
| 134 | crossLimit = -fMargin.fLeft - fMargin.fRight; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 135 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 136 | mainGetSizeP = &SkView::height; |
| 137 | crossGetSizeP = &SkView::width; |
| 138 | mainLocP = &SkView::setLocY; |
| 139 | crossLocP = &SkView::setLocX; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 140 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 141 | mainSetSizeP = &SkView::setHeight; |
| 142 | crossSetSizeP = &SkView::setWidth; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 143 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 144 | flexMask = SkView::kFlexV_Mask; |
| 145 | } |
| 146 | crossLimit += (parent->*crossGetSizeP)(); |
| 147 | if (fAlign != kStretch_Align) |
| 148 | crossSetSizeP = NULL; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 149 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 150 | int childCount, flexCount; |
| 151 | SkScalar childLimit = compute_children_limit(parent, mainGetSizeP, &childCount, flexMask, &flexCount); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 152 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 153 | if (childCount == 0) |
| 154 | return; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 155 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 156 | childLimit += (childCount - 1) * fSpacer; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 157 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 158 | SkScalar parentLimit = (parent->*mainGetSizeP)() - startM - endM; |
| 159 | SkScalar pos = startM + gAlignProcs[fPack](childLimit, parentLimit); |
| 160 | SkScalar flexAmount = 0; |
| 161 | SkView::B2FIter iter(parent); |
| 162 | SkView* child; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 163 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 164 | if (flexCount > 0 && parentLimit > childLimit) |
| 165 | flexAmount = (parentLimit - childLimit) / flexCount; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 166 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 167 | while ((child = iter.next()) != NULL) |
| 168 | { |
| 169 | if (fRound) |
| 170 | pos = SkIntToScalar(SkScalarRound(pos)); |
| 171 | (child->*mainLocP)(pos); |
| 172 | SkScalar crossLoc = crossStartM + gAlignProcs[fAlign]((child->*crossGetSizeP)(), crossLimit); |
| 173 | if (fRound) |
| 174 | crossLoc = SkIntToScalar(SkScalarRound(crossLoc)); |
| 175 | (child->*crossLocP)(crossLoc); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 176 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 177 | if (crossSetSizeP) |
| 178 | (child->*crossSetSizeP)(crossLimit); |
| 179 | if (child->getFlags() & flexMask) |
| 180 | (child->*mainSetSizeP)(flexAmount); |
| 181 | pos += (child->*mainGetSizeP)() + fSpacer; |
| 182 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 183 | } |
| 184 | |
| 185 | ////////////////////////////////////////////////////////////////////////////////////// |
| 186 | |
| 187 | #ifdef SK_DEBUG |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 188 | static void assert_no_attr(const SkDOM& dom, const SkDOM::Node* node, const char attr[]) |
| 189 | { |
| 190 | const char* value = dom.findAttr(node, attr); |
| 191 | if (value) |
| 192 | SkDebugf("unknown attribute %s=\"%s\"\n", attr, value); |
| 193 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 194 | #else |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 195 | #define assert_no_attr(dom, node, attr) |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 196 | #endif |
| 197 | |
| 198 | void SkStackViewLayout::onInflate(const SkDOM& dom, const SkDOM::Node* node) |
| 199 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 200 | int index; |
| 201 | SkScalar value[4]; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 202 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 203 | if ((index = dom.findList(node, "orient", "horizontal,vertical")) >= 0) |
| 204 | this->setOrient((Orient)index); |
| 205 | else { |
| 206 | assert_no_attr(dom, node, "orient"); |
tomhudson@google.com | 2c2508d | 2011-07-29 13:44:30 +0000 | [diff] [blame] | 207 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 208 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 209 | if (dom.findScalars(node, "margin", value, 4)) |
| 210 | { |
| 211 | SkRect margin; |
| 212 | margin.set(value[0], value[1], value[2], value[3]); |
| 213 | this->setMargin(margin); |
| 214 | } |
| 215 | else { |
| 216 | assert_no_attr(dom, node, "margin"); |
tomhudson@google.com | 2c2508d | 2011-07-29 13:44:30 +0000 | [diff] [blame] | 217 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 218 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 219 | if (dom.findScalar(node, "spacer", value)) |
| 220 | this->setSpacer(value[0]); |
| 221 | else { |
| 222 | assert_no_attr(dom, node, "spacer"); |
tomhudson@google.com | 2c2508d | 2011-07-29 13:44:30 +0000 | [diff] [blame] | 223 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 224 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 225 | if ((index = dom.findList(node, "pack", "start,center,end")) >= 0) |
| 226 | this->setPack((Pack)index); |
| 227 | else { |
| 228 | assert_no_attr(dom, node, "pack"); |
tomhudson@google.com | 2c2508d | 2011-07-29 13:44:30 +0000 | [diff] [blame] | 229 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 230 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 231 | if ((index = dom.findList(node, "align", "start,center,end,stretch")) >= 0) |
| 232 | this->setAlign((Align)index); |
| 233 | else { |
| 234 | assert_no_attr(dom, node, "align"); |
tomhudson@google.com | 2c2508d | 2011-07-29 13:44:30 +0000 | [diff] [blame] | 235 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 236 | } |
| 237 | |
| 238 | /////////////////////////////////////////////////////////////////////////////////////////// |
| 239 | |
| 240 | SkFillViewLayout::SkFillViewLayout() |
| 241 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 242 | fMargin.setEmpty(); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 243 | } |
| 244 | |
| 245 | void SkFillViewLayout::getMargin(SkRect* r) const |
| 246 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 247 | if (r) |
| 248 | *r = fMargin; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 249 | } |
| 250 | |
| 251 | void SkFillViewLayout::setMargin(const SkRect& margin) |
| 252 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 253 | fMargin = margin; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 254 | } |
| 255 | |
| 256 | void SkFillViewLayout::onLayoutChildren(SkView* parent) |
| 257 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 258 | SkView::B2FIter iter(parent); |
| 259 | SkView* child; |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 260 | |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 261 | while ((child = iter.next()) != NULL) |
| 262 | { |
| 263 | child->setLoc(fMargin.fLeft, fMargin.fTop); |
| 264 | child->setSize( parent->width() - fMargin.fRight - fMargin.fLeft, |
| 265 | parent->height() - fMargin.fBottom - fMargin.fTop); |
| 266 | } |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 267 | } |
| 268 | |
| 269 | void SkFillViewLayout::onInflate(const SkDOM& dom, const SkDOM::Node* node) |
| 270 | { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 271 | this->INHERITED::onInflate(dom, node); |
| 272 | (void)dom.findScalars(node, "margin", (SkScalar*)&fMargin, 4); |
reed@android.com | 8a1c16f | 2008-12-17 15:59:43 +0000 | [diff] [blame] | 273 | } |