2016-05-25 18:56:41 -07:00
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2011 The Android Open Source Project
|
|
|
|
|
*
|
|
|
|
|
* Licensed 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
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
package com.android.launcher3;
|
|
|
|
|
|
2018-01-18 17:14:05 -08:00
|
|
|
import static com.android.launcher3.ButtonDropTarget.TOOLTIP_DEFAULT;
|
2018-03-14 17:51:49 -07:00
|
|
|
import static com.android.launcher3.anim.AlphaUpdateListener.updateVisibility;
|
2017-10-24 14:54:30 -07:00
|
|
|
|
2016-05-25 18:56:41 -07:00
|
|
|
import android.animation.TimeInterpolator;
|
|
|
|
|
import android.content.Context;
|
2018-01-08 14:19:34 -08:00
|
|
|
import android.graphics.Rect;
|
2016-05-25 18:56:41 -07:00
|
|
|
import android.util.AttributeSet;
|
2021-09-01 12:33:00 -07:00
|
|
|
import android.util.Log;
|
2021-04-29 00:36:06 +01:00
|
|
|
import android.util.TypedValue;
|
2018-01-08 14:19:34 -08:00
|
|
|
import android.view.Gravity;
|
2016-05-25 18:56:41 -07:00
|
|
|
import android.view.View;
|
|
|
|
|
import android.view.ViewDebug;
|
|
|
|
|
import android.view.ViewPropertyAnimator;
|
2018-01-08 14:19:34 -08:00
|
|
|
import android.widget.FrameLayout;
|
2016-05-25 18:56:41 -07:00
|
|
|
|
2021-09-01 12:33:00 -07:00
|
|
|
import androidx.annotation.NonNull;
|
|
|
|
|
|
2017-10-26 15:36:10 -07:00
|
|
|
import com.android.launcher3.anim.Interpolators;
|
2016-05-25 18:56:41 -07:00
|
|
|
import com.android.launcher3.dragndrop.DragController;
|
2018-01-08 14:19:34 -08:00
|
|
|
import com.android.launcher3.dragndrop.DragController.DragListener;
|
2016-08-16 15:36:48 -07:00
|
|
|
import com.android.launcher3.dragndrop.DragOptions;
|
2021-09-01 12:33:00 -07:00
|
|
|
import com.android.launcher3.testing.TestProtocol;
|
2016-05-25 18:56:41 -07:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
import java.util.Arrays;
|
|
|
|
|
|
2016-05-25 18:56:41 -07:00
|
|
|
/*
|
|
|
|
|
* The top bar containing various drop targets: Delete/App Info/Uninstall.
|
|
|
|
|
*/
|
2018-01-18 17:14:05 -08:00
|
|
|
public class DropTargetBar extends FrameLayout
|
2018-01-08 14:19:34 -08:00
|
|
|
implements DragListener, Insettable {
|
2016-05-25 18:56:41 -07:00
|
|
|
|
|
|
|
|
protected static final int DEFAULT_DRAG_FADE_DURATION = 175;
|
2017-10-26 15:36:10 -07:00
|
|
|
protected static final TimeInterpolator DEFAULT_INTERPOLATOR = Interpolators.ACCEL;
|
2016-05-25 18:56:41 -07:00
|
|
|
|
2018-01-08 14:19:34 -08:00
|
|
|
private final Runnable mFadeAnimationEndRunnable =
|
2018-03-27 13:44:00 -07:00
|
|
|
() -> updateVisibility(DropTargetBar.this);
|
2016-05-25 18:56:41 -07:00
|
|
|
|
|
|
|
|
@ViewDebug.ExportedProperty(category = "launcher")
|
|
|
|
|
protected boolean mDeferOnDragEnd;
|
|
|
|
|
|
|
|
|
|
@ViewDebug.ExportedProperty(category = "launcher")
|
|
|
|
|
protected boolean mVisible = false;
|
|
|
|
|
|
2017-10-24 14:54:30 -07:00
|
|
|
private ButtonDropTarget[] mDropTargets;
|
2016-05-25 18:56:41 -07:00
|
|
|
private ViewPropertyAnimator mCurrentAnimation;
|
|
|
|
|
|
2018-01-18 17:14:05 -08:00
|
|
|
private boolean mIsVertical = true;
|
|
|
|
|
|
2016-05-25 18:56:41 -07:00
|
|
|
public DropTargetBar(Context context, AttributeSet attrs) {
|
|
|
|
|
super(context, attrs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public DropTargetBar(Context context, AttributeSet attrs, int defStyle) {
|
|
|
|
|
super(context, attrs, defStyle);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onFinishInflate() {
|
|
|
|
|
super.onFinishInflate();
|
2018-01-18 17:14:05 -08:00
|
|
|
mDropTargets = new ButtonDropTarget[getChildCount()];
|
|
|
|
|
for (int i = 0; i < mDropTargets.length; i++) {
|
|
|
|
|
mDropTargets[i] = (ButtonDropTarget) getChildAt(i);
|
|
|
|
|
mDropTargets[i].setDropTargetBar(this);
|
|
|
|
|
}
|
2016-05-25 18:56:41 -07:00
|
|
|
}
|
|
|
|
|
|
2018-01-08 14:19:34 -08:00
|
|
|
@Override
|
|
|
|
|
public void setInsets(Rect insets) {
|
|
|
|
|
FrameLayout.LayoutParams lp = (FrameLayout.LayoutParams) getLayoutParams();
|
2022-05-20 15:21:48 +00:00
|
|
|
DeviceProfile grid = Launcher.getLauncher(getContext()).getDeviceProfile();
|
2018-01-18 17:14:05 -08:00
|
|
|
mIsVertical = grid.isVerticalBarLayout();
|
2018-01-08 14:19:34 -08:00
|
|
|
|
|
|
|
|
lp.leftMargin = insets.left;
|
|
|
|
|
lp.topMargin = insets.top;
|
|
|
|
|
lp.bottomMargin = insets.bottom;
|
|
|
|
|
lp.rightMargin = insets.right;
|
2018-01-18 17:14:05 -08:00
|
|
|
int tooltipLocation = TOOLTIP_DEFAULT;
|
2018-01-08 14:19:34 -08:00
|
|
|
|
2022-04-05 19:11:26 +01:00
|
|
|
int horizontalMargin;
|
|
|
|
|
if (grid.isTablet) {
|
|
|
|
|
// XXX: If the icon size changes across orientations, we will have to take
|
|
|
|
|
// that into account here too.
|
|
|
|
|
horizontalMargin = ((grid.widthPx - 2 * grid.edgeMarginPx
|
|
|
|
|
- (grid.inv.numColumns * grid.cellWidthPx))
|
|
|
|
|
/ (2 * (grid.inv.numColumns + 1)))
|
|
|
|
|
+ grid.edgeMarginPx;
|
2018-01-08 14:19:34 -08:00
|
|
|
} else {
|
2022-04-05 19:11:26 +01:00
|
|
|
horizontalMargin = getContext().getResources()
|
|
|
|
|
.getDimensionPixelSize(R.dimen.drop_target_bar_margin_horizontal);
|
|
|
|
|
}
|
|
|
|
|
lp.topMargin += grid.dropTargetBarTopMarginPx;
|
|
|
|
|
lp.bottomMargin += grid.dropTargetBarBottomMarginPx;
|
|
|
|
|
lp.width = grid.availableWidthPx - 2 * horizontalMargin;
|
|
|
|
|
if (mIsVertical) {
|
|
|
|
|
lp.leftMargin = (grid.widthPx - lp.width) / 2;
|
|
|
|
|
lp.rightMargin = (grid.widthPx - lp.width) / 2;
|
2022-03-25 16:07:46 +00:00
|
|
|
}
|
2022-04-05 19:11:26 +01:00
|
|
|
lp.height = grid.dropTargetBarSizePx;
|
|
|
|
|
lp.gravity = Gravity.CENTER_HORIZONTAL | Gravity.TOP;
|
|
|
|
|
|
2018-01-08 14:19:34 -08:00
|
|
|
setLayoutParams(lp);
|
2018-01-18 17:14:05 -08:00
|
|
|
for (ButtonDropTarget button : mDropTargets) {
|
2021-04-29 00:36:06 +01:00
|
|
|
button.setTextSize(TypedValue.COMPLEX_UNIT_PX, grid.dropTargetTextSizePx);
|
2018-01-18 17:14:05 -08:00
|
|
|
button.setToolTipLocation(tooltipLocation);
|
|
|
|
|
}
|
2018-01-08 14:19:34 -08:00
|
|
|
}
|
|
|
|
|
|
2016-05-25 18:56:41 -07:00
|
|
|
public void setup(DragController dragController) {
|
|
|
|
|
dragController.addDragListener(this);
|
2017-10-24 14:54:30 -07:00
|
|
|
for (int i = 0; i < mDropTargets.length; i++) {
|
|
|
|
|
dragController.addDragListener(mDropTargets[i]);
|
|
|
|
|
dragController.addDropTarget(mDropTargets[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-21 15:31:51 -07:00
|
|
|
@Override
|
|
|
|
|
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
|
2018-01-18 17:14:05 -08:00
|
|
|
int width = MeasureSpec.getSize(widthMeasureSpec);
|
|
|
|
|
int height = MeasureSpec.getSize(heightMeasureSpec);
|
|
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
int visibleCount = getVisibleButtonsCount();
|
|
|
|
|
if (visibleCount > 0) {
|
|
|
|
|
int availableWidth = width / visibleCount;
|
|
|
|
|
boolean textVisible = true;
|
|
|
|
|
boolean textResized = false;
|
|
|
|
|
float textSize = mDropTargets[0].getTextSize();
|
|
|
|
|
for (ButtonDropTarget button : mDropTargets) {
|
|
|
|
|
if (button.getVisibility() == GONE) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (button.isTextTruncated(availableWidth)) {
|
|
|
|
|
textSize = Math.min(textSize, button.resizeTextToFit(availableWidth));
|
|
|
|
|
textResized = true;
|
|
|
|
|
}
|
|
|
|
|
textVisible = textVisible && !button.isTextTruncated(availableWidth);
|
|
|
|
|
}
|
2017-08-21 15:31:51 -07:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
if (textResized) {
|
|
|
|
|
for (ButtonDropTarget button : mDropTargets) {
|
|
|
|
|
button.setTextSize(textSize);
|
2022-04-25 13:50:28 +01:00
|
|
|
}
|
2022-05-20 15:21:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int widthSpec = MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST);
|
|
|
|
|
int heightSpec = MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY);
|
|
|
|
|
for (ButtonDropTarget button : mDropTargets) {
|
|
|
|
|
if (button.getVisibility() != GONE) {
|
|
|
|
|
button.setTextVisible(textVisible);
|
|
|
|
|
button.measure(widthSpec, heightSpec);
|
2018-01-18 17:14:05 -08:00
|
|
|
}
|
|
|
|
|
}
|
2017-08-21 15:31:51 -07:00
|
|
|
}
|
2018-01-18 17:14:05 -08:00
|
|
|
setMeasuredDimension(width, height);
|
2017-08-21 15:31:51 -07:00
|
|
|
}
|
|
|
|
|
|
2018-01-18 17:14:05 -08:00
|
|
|
@Override
|
|
|
|
|
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
|
2022-05-20 15:21:48 +00:00
|
|
|
int visibleCount = getVisibleButtonsCount();
|
2021-02-02 13:45:34 -08:00
|
|
|
if (visibleCount == 0) {
|
2022-04-05 19:11:26 +01:00
|
|
|
return;
|
|
|
|
|
}
|
2018-01-18 17:14:05 -08:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
Launcher launcher = Launcher.getLauncher(getContext());
|
|
|
|
|
Workspace<?> workspace = launcher.getWorkspace();
|
|
|
|
|
DeviceProfile dp = launcher.getDeviceProfile();
|
|
|
|
|
int buttonHorizontalPadding = dp.dropTargetHorizontalPaddingPx;
|
|
|
|
|
int buttonVerticalPadding = dp.dropTargetVerticalPaddingPx;
|
2022-04-05 19:11:26 +01:00
|
|
|
int barCenter = (right - left) / 2;
|
2022-05-20 15:21:48 +00:00
|
|
|
|
|
|
|
|
ButtonDropTarget[] visibleButtons = Arrays.stream(mDropTargets)
|
|
|
|
|
.filter(b -> b.getVisibility() != GONE)
|
|
|
|
|
.toArray(ButtonDropTarget[]::new);
|
|
|
|
|
Arrays.stream(visibleButtons).forEach(
|
|
|
|
|
b -> b.setPadding(buttonHorizontalPadding, buttonVerticalPadding,
|
|
|
|
|
buttonHorizontalPadding, buttonVerticalPadding));
|
2022-04-05 19:11:26 +01:00
|
|
|
|
|
|
|
|
if (visibleCount == 1) {
|
2022-05-20 15:21:48 +00:00
|
|
|
ButtonDropTarget button = visibleButtons[0];
|
2022-04-05 19:11:26 +01:00
|
|
|
button.layout(barCenter - (button.getMeasuredWidth() / 2), 0,
|
|
|
|
|
barCenter + (button.getMeasuredWidth() / 2), button.getMeasuredHeight());
|
|
|
|
|
} else if (visibleCount == 2) {
|
|
|
|
|
int buttonGap = dp.dropTargetGapPx;
|
|
|
|
|
|
|
|
|
|
if (dp.isTwoPanels) {
|
2022-05-20 15:21:48 +00:00
|
|
|
ButtonDropTarget leftButton = visibleButtons[0];
|
2022-04-05 19:11:26 +01:00
|
|
|
leftButton.layout(barCenter - leftButton.getMeasuredWidth() - (buttonGap / 2), 0,
|
|
|
|
|
barCenter - (buttonGap / 2), leftButton.getMeasuredHeight());
|
|
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
ButtonDropTarget rightButton = visibleButtons[1];
|
2022-04-05 19:11:26 +01:00
|
|
|
rightButton.layout(barCenter + (buttonGap / 2), 0,
|
2022-05-20 15:21:48 +00:00
|
|
|
barCenter + rightButton.getMeasuredWidth() + (buttonGap / 2),
|
2022-04-05 19:11:26 +01:00
|
|
|
rightButton.getMeasuredHeight());
|
2022-05-20 15:21:48 +00:00
|
|
|
} else if (dp.isTablet) {
|
|
|
|
|
int numberOfMargins = visibleCount - 1;
|
|
|
|
|
int buttonWidths = Arrays.stream(mDropTargets)
|
|
|
|
|
.filter(b -> b.getVisibility() != GONE)
|
|
|
|
|
.mapToInt(ButtonDropTarget::getMeasuredWidth)
|
|
|
|
|
.sum();
|
|
|
|
|
int totalWidth = buttonWidths + (numberOfMargins * buttonGap);
|
|
|
|
|
int buttonsStartMargin = barCenter - (totalWidth / 2);
|
|
|
|
|
|
|
|
|
|
int start = buttonsStartMargin;
|
|
|
|
|
for (ButtonDropTarget button : visibleButtons) {
|
|
|
|
|
int margin = (start != buttonsStartMargin) ? buttonGap : 0;
|
|
|
|
|
button.layout(start + margin, 0, start + margin + button.getMeasuredWidth(),
|
|
|
|
|
button.getMeasuredHeight());
|
|
|
|
|
start += button.getMeasuredWidth() + margin;
|
2018-01-18 17:14:05 -08:00
|
|
|
}
|
2022-05-20 15:21:48 +00:00
|
|
|
} else if (mIsVertical) {
|
|
|
|
|
// Center buttons over workspace, not screen.
|
|
|
|
|
int verticalCenter = (workspace.getRight() - workspace.getLeft()) / 2;
|
|
|
|
|
ButtonDropTarget leftButton = visibleButtons[0];
|
|
|
|
|
leftButton.layout(verticalCenter - leftButton.getMeasuredWidth() - (buttonGap / 2),
|
|
|
|
|
0, verticalCenter - (buttonGap / 2), leftButton.getMeasuredHeight());
|
|
|
|
|
|
|
|
|
|
ButtonDropTarget rightButton = visibleButtons[1];
|
|
|
|
|
rightButton.layout(verticalCenter + (buttonGap / 2), 0,
|
|
|
|
|
verticalCenter + rightButton.getMeasuredWidth() + (buttonGap / 2),
|
|
|
|
|
rightButton.getMeasuredHeight());
|
|
|
|
|
} else if (dp.isPhone) {
|
|
|
|
|
// Buttons aligned to outer edges of scaled workspace.
|
|
|
|
|
float scale = dp.getWorkspaceSpringLoadScale();
|
2022-04-22 11:29:17 +01:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
int workspaceWidth = (int) (launcher.getWorkspace().getNormalChildWidth() * scale);
|
|
|
|
|
int start = barCenter - (workspaceWidth / 2);
|
|
|
|
|
int end = barCenter + (workspaceWidth / 2);
|
2022-04-05 19:11:26 +01:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
ButtonDropTarget leftButton = visibleButtons[0];
|
|
|
|
|
ButtonDropTarget rightButton = visibleButtons[1];
|
2022-05-19 18:48:14 +01:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
// If the text within the buttons is too long, the buttons can overlap
|
|
|
|
|
int overlap = start + leftButton.getMeasuredWidth() + rightButton.getMeasuredWidth()
|
|
|
|
|
- end;
|
|
|
|
|
if (overlap > 0) {
|
|
|
|
|
end += overlap;
|
|
|
|
|
}
|
2022-05-19 18:48:14 +01:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
leftButton.layout(start, 0, start + leftButton.getMeasuredWidth(),
|
2022-04-05 19:11:26 +01:00
|
|
|
leftButton.getMeasuredHeight());
|
2022-05-20 15:21:48 +00:00
|
|
|
rightButton.layout(end - rightButton.getMeasuredWidth(), 0, end,
|
2022-04-05 19:11:26 +01:00
|
|
|
rightButton.getMeasuredHeight());
|
2017-08-21 15:31:51 -07:00
|
|
|
}
|
|
|
|
|
}
|
2018-01-18 17:14:05 -08:00
|
|
|
}
|
2017-08-21 15:31:51 -07:00
|
|
|
|
2022-05-20 15:21:48 +00:00
|
|
|
private int getVisibleButtonsCount() {
|
2018-01-18 17:14:05 -08:00
|
|
|
int visibleCount = 0;
|
2022-05-20 15:21:48 +00:00
|
|
|
for (ButtonDropTarget buttons : mDropTargets) {
|
|
|
|
|
if (buttons.getVisibility() != GONE) {
|
2018-01-18 17:14:05 -08:00
|
|
|
visibleCount++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return visibleCount;
|
2017-08-21 15:31:51 -07:00
|
|
|
}
|
|
|
|
|
|
2019-05-01 16:16:53 -07:00
|
|
|
public void animateToVisibility(boolean isVisible) {
|
2021-09-01 12:33:00 -07:00
|
|
|
if (TestProtocol.sDebugTracing) {
|
|
|
|
|
Log.d(TestProtocol.NO_DROP_TARGET, "8");
|
|
|
|
|
}
|
2016-05-25 18:56:41 -07:00
|
|
|
if (mVisible != isVisible) {
|
|
|
|
|
mVisible = isVisible;
|
|
|
|
|
|
|
|
|
|
// Cancel any existing animation
|
|
|
|
|
if (mCurrentAnimation != null) {
|
|
|
|
|
mCurrentAnimation.cancel();
|
|
|
|
|
mCurrentAnimation = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
float finalAlpha = mVisible ? 1 : 0;
|
|
|
|
|
if (Float.compare(getAlpha(), finalAlpha) != 0) {
|
|
|
|
|
setVisibility(View.VISIBLE);
|
|
|
|
|
mCurrentAnimation = animate().alpha(finalAlpha)
|
|
|
|
|
.setInterpolator(DEFAULT_INTERPOLATOR)
|
|
|
|
|
.setDuration(DEFAULT_DRAG_FADE_DURATION)
|
|
|
|
|
.withEndAction(mFadeAnimationEndRunnable);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* DragController.DragListener implementation
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
2016-08-16 15:36:48 -07:00
|
|
|
public void onDragStart(DropTarget.DragObject dragObject, DragOptions options) {
|
2021-09-01 12:33:00 -07:00
|
|
|
if (TestProtocol.sDebugTracing) {
|
|
|
|
|
Log.d(TestProtocol.NO_DROP_TARGET, "7");
|
|
|
|
|
}
|
2016-05-25 18:56:41 -07:00
|
|
|
animateToVisibility(true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This is called to defer hiding the delete drop target until the drop animation has completed,
|
|
|
|
|
* instead of hiding immediately when the drag has ended.
|
|
|
|
|
*/
|
|
|
|
|
protected void deferOnDragEnd() {
|
|
|
|
|
mDeferOnDragEnd = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void onDragEnd() {
|
|
|
|
|
if (!mDeferOnDragEnd) {
|
|
|
|
|
animateToVisibility(false);
|
|
|
|
|
} else {
|
|
|
|
|
mDeferOnDragEnd = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-10-24 14:54:30 -07:00
|
|
|
|
|
|
|
|
public ButtonDropTarget[] getDropTargets() {
|
|
|
|
|
return mDropTargets;
|
|
|
|
|
}
|
2021-09-01 12:33:00 -07:00
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onVisibilityChanged(@NonNull View changedView, int visibility) {
|
|
|
|
|
super.onVisibilityChanged(changedView, visibility);
|
2021-10-26 19:57:49 +00:00
|
|
|
if (TestProtocol.sDebugTracing) {
|
|
|
|
|
if (visibility == VISIBLE) {
|
|
|
|
|
Log.d(TestProtocol.NO_DROP_TARGET, "9");
|
|
|
|
|
} else {
|
|
|
|
|
Log.d(TestProtocol.NO_DROP_TARGET, "Hiding drop target", new Exception());
|
|
|
|
|
}
|
2021-09-01 12:33:00 -07:00
|
|
|
}
|
|
|
|
|
}
|
2016-05-25 18:56:41 -07:00
|
|
|
}
|