如何在Android中创建类似iPhone的旋转选择器?

15

iPhone上默认的旋转控件比Android更好看。看起来至少有一个Android应用程序(UrbanSpoon)已经能够复制这个控件,而且非常棒:http://www.urbanspoon.com/android

有人有关于如何创建这个控件的想法吗?代码会很有帮助。


1
如果这对你很重要,我建议你进行悬赏。或者,如果你在需要它的公司工作,请联系Urbanspoon并看看是否可以从他们那里购买小部件。 - the_drow
如果我理解你的话正确的话,你可以尝试使用http://w2davids.wordpress.com/2010/09/28/advanced-android-ui-wheelpicker/。 - pengwang
1
我认为这就是你要找的: http://code.google.com/p/android-wheel/ - Albin
2个回答

25

如果您不需要轨迹球的支持,那么只需使用WebView和一些对现有JavaScript进行微小调整以模仿Apple的UIPickerView即可创建像这样的应用程序。

Fancy Spinner Image

Matteo Spinelli已经完成了大部分的工作,因此可以先下载他的代码,然后将这些更改应用于spinningwheel.js。他的代码希望从屏幕底部弹出选择器,并带有取消和完成按钮,因此我们需要修改一些行以消除此行为。

--- spinningwheel.js.orig   2010-05-26 00:17:00.411954051 -0700
+++ spinningwheel.js    2010-05-26 00:16:32.319010720 -0700
@@ -67,12 +67,10 @@
 
    onOrientationChange: function (e) {
        window.scrollTo(0, 0);
-       this.swWrapper.style.top = window.innerHeight + window.pageYOffset + 'px';
        this.calculateSlotsWidth();
    },
    
    onScroll: function (e) {
-       this.swWrapper.style.top = window.innerHeight + window.pageYOffset + 'px';
    },
 
    lockScreen: function (e) {
@@ -113,9 +111,9 @@
        // Create the Spinning Wheel main wrapper
        div = document.createElement('div');
        div.id = 'sw-wrapper';
-       div.style.top = window.innerHeight + window.pageYOffset + 'px';     // Place the SW down the actual viewing screen
+       div.style.top = 0;
        div.style.webkitTransitionProperty = '-webkit-transform';
-       div.innerHTML = '<div id="sw-header"><div id="sw-cancel">Cancel</' + 'div><div id="sw-done">Done</' + 'div></' + 'div><div id="sw-slots-wrapper"><div id="sw-slots"></' + 'div></' + 'div><div id="sw-frame"></' + 'div>';
+       div.innerHTML = '<div id="sw-slots-wrapper"><div id="sw-slots"></' + 'div></' + 'div><div id="sw-frame"></' + 'div>';
 
        document.body.appendChild(div);
 
@@ -164,8 +162,6 @@
        window.addEventListener('scroll', this, true);              // Reposition SW on page scroll
 
        // Cancel/Done buttons events
-       document.getElementById('sw-cancel').addEventListener('touchstart', this, false);
-       document.getElementById('sw-done').addEventListener('touchstart', this, false);
 
        // Add scrolling to the slots
        this.swFrame.addEventListener('touchstart', this, false);
@@ -174,9 +170,6 @@
    open: function () {
        this.create();
 
-       this.swWrapper.style.webkitTransitionTimingFunction = 'ease-out';
-       this.swWrapper.style.webkitTransitionDuration = '400ms';
-       this.swWrapper.style.webkitTransform = 'translate3d(0, -260px, 0)';
    },
    
    
@@ -191,8 +184,6 @@
 
        this.swFrame.removeEventListener('touchstart', this, false);
 
-       document.getElementById('sw-cancel').removeEventListener('touchstart', this, false);
-       document.getElementById('sw-done').removeEventListener('touchstart', this, false);
 
        document.removeEventListener('touchstart', this, false);
        document.removeEventListener('touchmove', this, false);

此外,他提供的 index.html 并不完全符合您的要求,因此请使用此文件替换它,然后将 html、css、js 和 png 文件复制到您项目的 assets 目录中。

<html>
 <head>
  <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  <link rel="stylesheet" href="spinningwheel.css" type="text/css" media="all" />
  <script type="text/javascript" src="spinningwheel.js?v=1.4"></script>
  <title></title>
  <script type="text/javascript">
        
   function getData() {
    var results = SpinningWheel.getSelectedValues();
    window.android.sendResults(results.values.join(' ')  );
   }
            
   function notifyAndroid() {
    window.android.readyForJavascript('');
   }
        
  </script>
 </head>
 <body onload="javascript:notifyAndroid()"></body>
</html>
创建一个Activity,在其中启用WebView中的JavaScript,并添加回调函数以返回所选内容。
public class FancySpinner extends Activity {
    private WebView webView;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        ((Button) findViewById(R.id.GetSelectedTimeButton))
                .setOnClickListener(new OnClickListener() {
                    @Override
                    public void onClick(View v) {
                        webView.loadUrl("javascript:getData()");
                    }
                });
        webView = (WebView) findViewById(R.id.WebView01);
        webView.getSettings().setJavaScriptEnabled(true);
        webView.addJavascriptInterface(new AndroidBridge(), "android");
        webView.setVerticalScrollBarEnabled(false);
        webView.loadUrl("file:///android_asset/index.html");
    }
    private class AndroidBridge {
        public void sendResults(final String arg) {
            Toast.makeText(FancySpinner.this, arg, Toast.LENGTH_SHORT).show();
        }
        public void readyForJavascript(final String arg) {
            webView.loadUrl("javascript:SpinningWheel.addSlot({ " +
                    "1: 1, 2: 2, 3: 3,  4:  4,  5:  5,  6:  6," +
                    "7: 7, 8: 8, 9: 9, 10: 10, 11: 11, 12: 12 });");
            webView.loadUrl("javascript:SpinningWheel.addSlot({ " +
                    "1: 'AM', 2: 'PM'});");
            webView.loadUrl("javascript:SpinningWheel.open();");
        }
    }
}

最后,修改您的布局,使其具有高度适当设置的WebView。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" 
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <TextView 
        android:layout_width="fill_parent"
        android:layout_height="wrap_content" 
        android:text="@string/hello" />
    <WebView 
        android:id="@+id/WebView01" 
        android:background="#77CC0000"
        android:layout_height="215dp" 
        android:layout_width="fill_parent"
        android:focusable="false" />
    <Button 
        android:text="Read Selected Time" 
        android:id="@+id/GetSelectedTimeButton" 
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" />
</LinearLayout>

轮子没有转动!你可能需要修改你的JavaScript代码。 - Tawani
@Tawani,我不确定为什么它对你不起作用。也许你应该在Android浏览器应用程序中查看Matteo的演示页面,看看你的触摸事件是否会导致轮子旋转。 - Tim Kryger
1
它在2.1版本中可以运行,但在1.6版本中无法运行。 - JonF
JonF,您能否给我这段代码,我无法运行它,谢谢 wp19831014@gmail.com - pengwang
哇,这真的很棒!这正是我一直在寻找的! - Ramesh Akula

1
从截图看,它似乎只是一个带有特殊样式的列表视图,使其看起来是圆形的。在行为方面,我认为它与将中间项视为所选值的ListView没有任何区别。

我认为那不正确。如果您使用小部件,它实际上更像是三个ListView组合在一起。只是想知道有什么想法可以实现这一点... - greg7gkb
是的,抱歉我想表达的意思就是这样。每个单独的Spinner都是一个ListView,所以那张图片上有三个。 - Cheryl Simon

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接