- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
用户向左或向右滚动“磁带”以选择值(显示在中间框中)。磁带有最大值和最小值,达到最大值和最小值后将显示典型的过度滚动动画(在 Android 上发光;在 iOS 上反弹)。
Hixie 在 Gitter 上建议我可以只使用 GestureDetector
+ CustomPaint
,但我觉得我必须自己实现滚动逻辑并且不会利用 Flutter 的滑动和滚动实现。
编辑:经过进一步调查,我改变了我原来的方法,即使用低级小部件,例如 Scrollable
和 Viewport
。
我已经能够通过扩展 CustomPaint
并将其宽度设置为磁带的全长来创建磁带:_width = (_maxValue - _minValue) * _spacing;
然后我将自定义小部件放入 CustomScrollView 中:
import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';
void main() {
runApp(new MaterialApp(home: new Scaffold(
appBar: new AppBar(title: new Text("Test"),),
body: new CustomScrollView(
scrollDirection: Axis.horizontal,
slivers: <Widget>[
new SliverToBoxAdapter(
child: new Tape(),
)
],
)
)));
}
const _width = (_maxValue - _minValue) * spacing;
const spacing = 20.0;
const _minValue = 0;
const _maxValue = 100;
class Tape extends CustomPaint {
Tape() : super(
size: new Size(_width, 60.0),
painter: new _TapePainter(),
);
}
class _TapePainter extends CustomPainter {
Paint _tickPaint;
_TapePainter() {
_tickPaint = new Paint();
_tickPaint.color = Colors.black;
_tickPaint.strokeWidth = 1.0;
}
@override
void paint(Canvas canvas, Size size) {
var rect = Offset.zero & size;
var o1 = new Offset(0.0, 0.0);
var o2 = new Offset(0.0, rect.height);
while (o1.dx < size.width) {
canvas.drawLine(o1, o2, _tickPaint);
o1 = o1.translate(spacing, 0.0);
o2 = o2.translate(spacing, 0.0);
}
}
@override
bool shouldRepaint(_TapePainter oldDelegate) {
return true;
}
}
这实现了我想要的效果:我现在可以左右滚动磁带,并且免费获得过度滚动效果。
问题在于当前代码效率低下:整个磁带被绘制一次,滚动条只是在缓冲的位图中移动。这会导致非常大的“磁带”出现问题。
相反,我正在寻找的是在每一帧上重新绘制小部件,以便只需要计算和绘制可见部分。这也将允许我实现其他滚动相关的效果,例如当数字接近中心时动态淡入。
最佳答案
经过大量调查,我设法解决了这个问题。我很确定我的解决方案不是执行此操作的最佳方法,但它确实有效。如果有人可以对解决方案的质量以及如何改进它发表评论,我将不胜感激。
我从 SliverBoxAdapter
复制代码以返回 RenderSliverToBoxAdapter
的自定义版本,它在每个布局过程中公开可见的几何图形(小部件的实际可见部分) .然后,我的 CustomPainter
使用此信息将绘图命令限制为仅显示在可见区域内的命令。
请注意,下面的代码旨在作为概念验证,因此很难看。我将在此处将其扩展为成熟的解决方案:https://github.com/cachapa/FlutterTapeSelector
import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';
void main() {
runApp(new MaterialApp(
home: new Scaffold(
appBar: new AppBar(
title: new Text("Test"),
),
body: new CustomScrollView(
scrollDirection: Axis.horizontal,
slivers: <Widget>[
new CustomSliverToBoxAdapter(
child: new Tape(),
)
],
))));
}
const _width = (_maxValue - _minValue) * spacing;
const spacing = 20.0;
const _minValue = 0;
const _maxValue = 100;
class Tape extends CustomPaint {
Tape()
: super(
size: new Size(_width, 60.0),
painter: new _TapePainter(),
);
}
class _TapePainter extends CustomPainter {
Paint _tickPaint = new Paint();
_TapePainter() {
_tickPaint.color = Colors.black;
_tickPaint.strokeWidth = 2.0;
}
@override
void paint(Canvas canvas, Size size) {
var rect = Offset.zero & size;
// Extend drawing window to compensate for element sizes - avoids lines at either end "popping" into existence
var extend = _tickPaint.strokeWidth / 2.0;
// Calculate from which Tick we should start drawing
var tick = ((_visibleRect.left - extend) / spacing).ceil();
var startOffset = tick * spacing;
var o1 = new Offset(startOffset, 0.0);
var o2 = new Offset(startOffset, rect.height);
while (o1.dx < _visibleRect.right + extend) {
canvas.drawLine(o1, o2, _tickPaint);
o1 = o1.translate(spacing, 0.0);
o2 = o2.translate(spacing, 0.0);
}
}
@override
bool shouldRepaint(_TapePainter oldDelegate) {
return false;
}
}
class CustomSliverToBoxAdapter extends SingleChildRenderObjectWidget {
const CustomSliverToBoxAdapter({
Key key,
Widget child,
})
: super(key: key, child: child);
@override
CustomRenderSliverToBoxAdapter createRenderObject(BuildContext context) =>
new CustomRenderSliverToBoxAdapter();
}
class CustomRenderSliverToBoxAdapter extends RenderSliverSingleBoxAdapter {
CustomRenderSliverToBoxAdapter({
RenderBox child,
})
: super(child: child);
@override
void performLayout() {
if (child == null) {
geometry = SliverGeometry.zero;
return;
}
child.layout(constraints.asBoxConstraints(), parentUsesSize: true);
double childExtent;
switch (constraints.axis) {
case Axis.horizontal:
childExtent = child.size.width;
break;
case Axis.vertical:
childExtent = child.size.height;
break;
}
assert(childExtent != null);
final double paintedChildSize =
calculatePaintOffset(constraints, from: 0.0, to: childExtent);
assert(paintedChildSize.isFinite);
assert(paintedChildSize >= 0.0);
geometry = new SliverGeometry(
scrollExtent: childExtent,
paintExtent: paintedChildSize,
maxPaintExtent: childExtent,
hitTestExtent: paintedChildSize,
hasVisualOverflow: childExtent > constraints.remainingPaintExtent ||
constraints.scrollOffset > 0.0,
);
setChildParentData(child, constraints, geometry);
// Expose geometry
_visibleRect = new Rect.fromLTWH(
constraints.scrollOffset, 0.0, geometry.paintExtent, child.size.height);
}
}
Rect _visibleRect = Rect.zero;
关于Flutter:如何创建自定义可滚动小部件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45298672/
我想向一些用户公开一个 Web 部件,但不是所有用户。如何在“添加 Web 部件”弹出窗口中显示或隐藏 Web 部件?我想通过代码来做到这一点,我希望使用 SharePoint 角色来实现这一点。 最
我无法创建两个以上的 StatusBar 部分: HWND hStatusBar = CreateWindowEx(0, STATUSCLASSNAME, "", WS_CHILD | WS_VISI
使用 SharePoint 2007,如何在编辑页面模式下允许将 CEWP 添加到“添加 Web 部件”对话框的选择菜单?目前,我只能添加公告、日历、链接、共享文档、任务,但我无法添加 CEWP。我可
哪个 Web 部件以及如何配置它以查看来自不同网站集的列表? 请注意,我不想查看页面,而是查看列表。例如,在单独的网站集下查看来自不同团队网站的公告。 预先感谢您的帮助。 最佳答案 Data Form
以下是我在 FeatureDeactivation 事件处理程序中添加的代码片段。我无法获得删除 System.Web.UI.WebControls.WebParts 类型的 webpart 的解决方
我一直在尝试跟踪来自以下方面的信息: Long URL clipped to stop breaking the page 和 http://msdn.microsoft.com/en-us/libr
我想创建一个自定义 Web 部件,它具有 1 个以上的筛选器 Web 部件,并且可以在运行时/设计时连接到报表查看器 Web 部件(集成模式)。 我为此搜索了很多,但找不到一种方法来让单个 Web 部
我正在尝试创建一个 Web 部件,使用户无需离开 AllItems.aspx 页面即可编辑项目。 Web 部件应具有与 EditForm.aspx 页面类似的功能。 我已经使用 ConnectionC
这些年发布的许多应用程序都有新的 GUI 部件。iTunes 或 Twitter.app 中垂直布局的最小、最大和关闭按钮(但最新的具有默认布局),Safari 和终端中的选项卡控件,GarageBa
在具有数据库依赖性的 WSS3 或 MOSS2007 中部署 Web 部件的最佳方法是什么? .wsp 是否应该包含创建数据库的代码,我应该将 .wsp 封装在另一个处理数据库创建的安装程序中,还是应
我在我们位于 http://sharepoint:12345 的 moss 服务器上创建了一个新的共享点站点并毫无问题地向其添加了 CQWP。 我有一个指向同一台服务器的域名。所以我指向了http:/
在官方 Office 2007 站点中有许多对筛选器 Web 部件的引用。当我尝试添加其中之一时,我的 Sharepoint 中的 Web 部件列表没有显示任何筛选器 Web 部件。 如果有人遇到相同
我被要求在 Sharepoint 中创建一个 Web 部件,列出用户在网站集中访问的最后 10 个文档。 我的客户想要一种快速的方式让用户访问文档,这样他们就不必翻遍文件夹结构来查找文档,因为大多数时
我需要使用 C# 以编程方式将 SharePoint Web 部件“站点用户”添加到页面。 我知道如何添加 Web 部件,但如何从 Share Point 获取“站点用户”Web 部件?我不知道如何实
我正在使用 MEF 在我的应用程序中加载插件。一切正常,但我希望在将新部件放入我的应用程序文件夹时发现它们。这可能吗? DirectoryCatalog 有一个 Changed 事件,但我不确定它是如
我有一个 Winforms 桌面应用程序正在加载具有相同接口(interface)类型的多个 MEF 部件。 问题:当我尝试加载多个相同类型时,出现以下异常: 组成保持不变。由于以下错误,更改被拒绝:
我有一个内容查询 Web 部件,它按内容类型对网站集进行查询。我已按内容类型对其进行了分组,因此我有: -- Agenda (Content Type) ----Agenda #1 ----Agend
考虑以下 SharePoint 站点层次结构: - Site Collection - Site1 - Subsite1 - AnotherSubsite1
好吧,在我的 SharePoint (2013) 网站中,我制作了一个简单的 JavaScript Web 部件,每五分钟刷新一次页面。我去调整时间,在刷新前输入等待时间的地方退格,然后不假思索地退出
我不知道 Sharepoint 脚本,我的同事也不知道 JavaScript。他使用了他在 http://www.wonderlaura.com/Lists/Posts/Post.aspx?ID=22
我是一名优秀的程序员,十分优秀!