Соединение макета Android с его логикой за пределами владельца - PullRequest
3 голосов
/ 20 марта 2012

Я сталкиваюсь с некоторыми сложностями при разработке приложения для Android средней сложности.Я ищу информацию о возможности использования code-behind -подобных техник для облегчения обслуживания программного обеспечения Android.

В настоящее время (пожалуйста, выделите что-то не так), я обнаружил, что для создания многошаговойМастер с дополнительными диалогами (например, диалоги, которые не являются частью основной последовательности) Мне нужно кодировать один файл макета XML с одним ViewFlipper, содержащий каждое подпредставление как дочерний узел.Сегодня я узнал, как перемещаться между представлениями больше, чем вперед / назад (viewFlipper.setDisplayedChild(i)), предоставляя доступ к дополнительным представлениям.

Теперь весь код Java содержится в основном классе Activity, который начинаетплохо выглядетьКак опытный разработчик .NET, я узнал, как использовать пользовательские элементы управления для обертывания макетов и бизнес-логики внутри модулей.

Я знаю , что в Android я могу программно определить представление как независимоекласс и добавить его к основному макету программно, однако я хочу знать, возможно ли в Android определить макет с помощью XML (для более легкого создания / редактирования WYSIWYG) и определить весь код в выделенном классе с логикой инициализации, обратными вызовами кнопок, асинхронные задачи и т. д.

Я не уверен, выполнимо ли это или есть хороший компромисс, который может быть достигнут.

Я прочитал этот вопрос без очисткимои сомнения.

Спасибо.

Примеры кода:

Извлечение файла макета (я ожидаю 4 шага мастера, представление справки и представление EULA)

<?xml version="1.0" encoding="utf-8"?>
<ViewFlipper xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/view_phone"
    style="@android:style/Theme.Light.NoTitleBar"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent" >

    <!-- First screen/welcome -->

    <LinearLayout
        android:id="@+id/view_phone_screen1"
        style="@android:style/Theme.Light.NoTitleBar"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:orientation="vertical"
        android:weightSum="100" >

        <TextView
            android:id="@+id/view_phone_screen1_lblChooseProvider"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:gravity="center_vertical|center_horizontal"
            android:text="@string/view_phone_lblChooseProvider_1ststep"
            android:textAppearance="?android:attr/textAppearanceLarge" />

        <ImageButton
            android:id="@+id/view_phone_btnFrecciarossa"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:clickable="true"
            android:contentDescription="@string/provider_FRECCIAROSSA"
            android:gravity="center_vertical|clip_vertical"
            android:padding="10dp"
            android:src="@drawable/logo_frecciarossa"
            android:tag="@+id/provider_FRECCIAROSSA" />

        <ImageButton
            android:id="@+id/view_phone_btnItalo"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:clickable="true"
            android:contentDescription="@string/provider_ITALO"
            android:gravity="center_vertical|clip_vertical"
            android:padding="10dp"
            android:src="@drawable/logo_italo"
            android:tag="@+id/provider_ITALO" />
    </LinearLayout>


    <!-- Second screen - will need to do some asynchronous task -->
    <RelativeLayout
        android:id="@+id/view_phone_screen2"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent" >

        <TextView
            android:id="@+id/view_phone_screen2_lblConnectingToWifi"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:gravity="center_vertical|center_horizontal"
            android:text="@string/view_phone_lblConnectToWifi_2ndstep"
            android:textAppearance="?android:attr/textAppearanceLarge" />


        <TextView
            android:id="@+id/view_phone_step2_lblConnectedToWifi"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_below="@+id/imageView1"
            android:layout_centerHorizontal="true"
            android:layout_marginTop="58dp"
            android:text="@string/view_phone_step2_connectingToWifi"
            android:textAppearance="?android:attr/textAppearanceLarge" />

        <TextView
            android:id="@+id/view_phone_step2_lblPhoneNumber"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_alignLeft="@+id/editText1"
            android:layout_below="@+id/view_phone_step2_lblConnectedToWifi"
            android:layout_marginTop="51dp"
            android:text="@string/view_phone_step2_msgInputPhoneNumber"
            android:textAppearance="?android:attr/textAppearanceMedium" />

        <TextView
            android:id="@+id/view_phone_step2_lblUnableDetectPhoneNumber"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_alignParentRight="true"
            android:text="@string/view_phone_step2_msgUnableDetectPhoneNumber"
            android:textAppearance="?android:attr/textAppearanceSmall"
            android:visibility="invisible" />

        <Button
            android:id="@+id/view_phone_screen2_backward"
            style="@style/buttonBackward" />

        <Button
            android:id="@+id/view_phone_screen2_forward"
            style="@style/buttonForward_disabled"
            android:enabled="false" />

        <EditText
            android:id="@+id/view_phone_step2_txtPhoneNumber"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_alignParentLeft="true"
            android:layout_alignRight="@+id/view_phone_step2_lblPhoneNumber"
            android:layout_below="@+id/view_phone_step2_lblPhoneNumber"
            android:inputType="phone"
            android:singleLine="true" >

            <requestFocus />
        </EditText>
    </RelativeLayout>

</ViewFlipper>

Пример кода из Activity (ожидаемо реализовать ВСЕ логику мастера 4 + 2 шага)

public class MyActivity extends Activity {
/** Called when the activity is first created. */

private final static String LOG_TAG = "LOG_TAG";
private int stepNumber;

@Override
public void onCreate(Bundle savedInstanceState) {
    this.requestWindowFeature(Window.FEATURE_NO_TITLE);
    this.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);

    super.onCreate(savedInstanceState);

    this.stepNumber=1;

    setContentView(R.layout.view_phone);

    //This class wraps the click for the two buttons
    ProviderSelectionListener providerSelectionListener = new ProviderSelectionListener(this);
    this.findViewById(R.id.view_phone_btnFrecciarossa).setOnClickListener(providerSelectionListener);
    this.findViewById(R.id.view_phone_btnItalo).setOnClickListener(providerSelectionListener);
}

@Override
protected void onPause() {
    super.onPause();

    try {
        if (MyApplication.getPlatformManager() != null)
            MyApplication.getPlatformManager().onApplicationPause();
    } catch (MyCustomException e) {
        // WTF (Worse Than Failure!)
        Log.e(LOG_TAG, super.getString(R.string.zf_error_unknown_error_pauseactivity), e);
        e.printStackTrace();
    }
}

@Override
protected void onResume() {
    super.onResume();

    try {
        if (MyApplication.getPlatformManager() != null)
            MyApplication.getPlatformManager().onApplicationResume();
    } catch (MyCustomException e) {
        // WTF (Worse Than Failure!)
        Log.e(LOG_TAG, super.getString(R.string.zf_error_unknown_error_pauseactivity), e);
        e.printStackTrace();
    }
}

/*
 * SLIDE INIZIO
 */
protected void slideNext() {
    ViewFlipper vf = (ViewFlipper) findViewById(R.id.view_phone);

    vf.setOutAnimation(getApplicationContext(), R.anim.slide_out_left);
    vf.setInAnimation(getApplicationContext(), R.anim.slide_in_right);
    vf.showNext();

}

protected void slidePrevious() {
    ViewFlipper vf = (ViewFlipper) findViewById(R.id.view_phone);

    vf.setOutAnimation(getApplicationContext(), R.anim.slide_out_right);
    vf.setInAnimation(getApplicationContext(), R.anim.slide_in_left);
    vf.showPrevious();
}

/*
 * SLIDE FINE
 */

/*
 * STEP 1 INIZIO
 */
public void completeStep1(ISmsWifiProvider provider) {
    if (provider == null) {
        Log.e(LOG_TAG, "Provider nullo");
        return;
    }

    MyApplication.setAuthenticationProvider(provider);

    slideNext();

    initializeStep2();
}

public void returnToStep1() {
    MyApplication.setAuthenticationProvider(null);
    slidePrevious();
}

/*
 * STEP 1 FINE
 */

/*
 * STEP 2 INIZIO
 */

private void initializeStep2() {
    // Event handler
    Button backButton = (Button) findViewById(R.id.view_phone_screen2_backward), fwButton = (Button) findViewById(R.id.view_phone_screen2_forward);
    fwButton.setEnabled(false);
    backButton.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
            returnToStep1();
        }
    });

}

/*
 * STEP 2 FINE
 */


@Override
public void onBackPressed() {
    // This will be called either automatically for you on 2.0
    // or later, or by the code above on earlier versions of the
    // platform.
    return;
}

}

1 Ответ

0 голосов
/ 20 марта 2012

Я хочу знать, возможно ли в Android определить макет с помощью XML (для упрощения создания / редактирования WYSIWYG) и определить весь код в выделенном классе с логикой инициализации, обратными вызовами кнопок, асинхронными задачами и т. Д.

Да. Это одна из техник для создания кастома View. Например, у меня есть пользовательский ColorMixer виджет в этом проекте , который можно использовать непосредственно в деятельности, в диалоге или в пользовательских настройках.

Я согласен с вашим тактическим решением реализовать волшебника с помощью ViewFlipper - см. этот ответ на другой вопрос StackOverflow для "Теории деятельности Мерфи".

Я подозреваю, что правильный ответ, в более долгосрочной перспективе, заключается в том, чтобы кто-то (кто может быть мной) придумал шаблон волшебника на основе Fragment, поскольку это дает вам развязку, которую вы желаете.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...