gov.wa.wsdot.mobile.client.activities.ferries.schedules.departures.FerriesRouteDeparturesViewGwtImpl.java Source code

Java tutorial

Introduction

Here is the source code for gov.wa.wsdot.mobile.client.activities.ferries.schedules.departures.FerriesRouteDeparturesViewGwtImpl.java

Source

/*
 * Copyright (c) 2016 Washington State Department of Transportation
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>
 *
 */

package gov.wa.wsdot.mobile.client.activities.ferries.schedules.departures;

import com.google.gwt.aria.client.Roles;
import com.google.gwt.aria.client.SelectedValue;
import com.google.gwt.core.client.GWT;
import com.google.gwt.event.dom.client.ChangeEvent;
import com.google.gwt.event.logical.shared.SelectionEvent;
import com.google.gwt.i18n.client.DateTimeFormat;
import com.google.gwt.i18n.client.TimeZone;
import com.google.gwt.i18n.client.TimeZoneInfo;
import com.google.gwt.i18n.client.constants.TimeZoneConstants;
import com.google.gwt.safehtml.shared.SafeHtml;
import com.google.gwt.safehtml.shared.SafeHtmlUtils;
import com.google.gwt.uibinder.client.UiBinder;
import com.google.gwt.uibinder.client.UiField;
import com.google.gwt.uibinder.client.UiHandler;
import com.google.gwt.user.client.ui.Composite;
import com.google.gwt.user.client.ui.HTML;
import com.google.gwt.user.client.ui.Widget;
import com.googlecode.mgwt.dom.client.event.tap.TapEvent;
import com.googlecode.mgwt.ui.client.MGWT;
import com.googlecode.mgwt.ui.client.widget.base.HasRefresh;
import com.googlecode.mgwt.ui.client.widget.header.HeaderTitle;
import com.googlecode.mgwt.ui.client.widget.input.listbox.MListBox;
import com.googlecode.mgwt.ui.client.widget.list.celllist.CellList;
import com.googlecode.mgwt.ui.client.widget.list.celllist.CellSelectedEvent;
import com.googlecode.mgwt.ui.client.widget.panel.flex.FlexSpacer;
import com.googlecode.mgwt.ui.client.widget.panel.flex.RootFlexPanel;
import com.googlecode.mgwt.ui.client.widget.panel.pull.PullArrowHeader;
import com.googlecode.mgwt.ui.client.widget.panel.pull.PullArrowWidget;
import com.googlecode.mgwt.ui.client.widget.panel.pull.PullPanel;
import com.googlecode.mgwt.ui.client.widget.panel.pull.PullPanel.Pullhandler;
import com.googlecode.mgwt.ui.client.widget.panel.scroll.ScrollPanel;
import com.googlecode.mgwt.ui.client.widget.progress.ProgressIndicator;
import com.googlecode.mgwt.ui.client.widget.tabbar.TabPanel;
import gov.wa.wsdot.mobile.client.activities.camera.CameraCell;
import gov.wa.wsdot.mobile.client.util.ParserUtils;
import gov.wa.wsdot.mobile.client.widget.button.image.BackImageButton;
import gov.wa.wsdot.mobile.client.widget.tabbar.CameraTabBarButton;
import gov.wa.wsdot.mobile.client.widget.tabbar.TimeTabBarButton;
import gov.wa.wsdot.mobile.shared.CameraItem;
import gov.wa.wsdot.mobile.shared.FerriesScheduleTimesItem;

import java.util.Date;
import java.util.List;

public class FerriesRouteDeparturesViewGwtImpl extends Composite implements FerriesRouteDeparturesView {

    /**
     * The UiBinder interface.
     */
    interface FerriesRouteDeparturesViewGwtImplUiBinder
            extends UiBinder<Widget, FerriesRouteDeparturesViewGwtImpl> {
    }

    /**
     * The UiBinder used to generate the view.
     */
    private static FerriesRouteDeparturesViewGwtImplUiBinder uiBinder = GWT
            .create(FerriesRouteDeparturesViewGwtImplUiBinder.class);

    @UiField
    HeaderTitle heading;

    @UiField
    RootFlexPanel times;

    @UiField
    RootFlexPanel cameras;

    @UiField(provided = true)
    CellList<FerriesScheduleTimesItem> cellList;

    @UiField(provided = true)
    CellList<CameraItem> cameraCellList;

    @UiField
    TabPanel tabPanel;

    @UiField
    TimeTabBarButton timesTab;

    @UiField
    CameraTabBarButton camerasTab;

    @UiField
    static ScrollPanel cameraScrollPanel;

    @UiField
    BackImageButton backButton;

    @UiField
    FlexSpacer leftFlexSpacer;

    @UiField(provided = true)
    PullPanel pullToRefresh;

    @UiField
    HTML title;

    @UiField
    ProgressIndicator progressIndicator;

    @UiField(provided = true)
    MListBox daysOfWeek;

    private Presenter presenter;
    private PullArrowHeader pullArrowHeader;
    private DateTimeFormat dateFormat = DateTimeFormat.getFormat("hh:mm a");
    private DateTimeFormat dayOfWeekFormat = DateTimeFormat.getFormat("EEEE");

    private final TimeZoneConstants timeZoneConstants = GWT.create(TimeZoneConstants.class);
    private final TimeZone usPacific = TimeZone
            .createTimeZone(TimeZoneInfo.buildTimeZoneData(timeZoneConstants.americaLosAngeles()));

    public FerriesRouteDeparturesViewGwtImpl() {

        pullToRefresh = new PullPanel();
        pullArrowHeader = new PullArrowHeader();
        pullToRefresh.setHeader(pullArrowHeader);

        daysOfWeek = new MListBox();

        handleOnLoad();

        cellList = new CellList<FerriesScheduleTimesItem>(
                new FerriesRouteDeparturesCell<FerriesScheduleTimesItem>() {

                    @Override
                    public String getDeparting(FerriesScheduleTimesItem model) {
                        Date departingTime = new Date(Long.parseLong(model.getDepartingTime()));

                        return dateFormat.format(departingTime, usPacific);
                    }

                    @Override
                    public String getArriving(FerriesScheduleTimesItem model) {
                        if (!model.getArrivingTime().equals("N/A")) {
                            Date arrivingTime = new Date(Long.parseLong(model.getArrivingTime()));

                            return dateFormat.format(arrivingTime, usPacific);
                        } else {
                            return "";
                        }
                    }

                    @Override
                    public SafeHtml getAnnotation(FerriesScheduleTimesItem model) {
                        if (model.getAnnotations() != null) {
                            return SafeHtmlUtils.fromTrustedString(model.getAnnotations());
                        } else {
                            return SafeHtmlUtils.fromString("");
                        }
                    }

                    @Override
                    public boolean canBeSelected(FerriesScheduleTimesItem model) {
                        return false;
                    }

                    @Override
                    public String getDriveUpSpaces(FerriesScheduleTimesItem model) {
                        return String.valueOf(model.getDriveUpSpaceCount());
                    }

                    @Override
                    public String getMaxSpaceCount(FerriesScheduleTimesItem model) {
                        return String.valueOf(model.getMaxSpaceCount());
                    }

                    @Override
                    public String getLastUpdated(FerriesScheduleTimesItem model) {
                        if (model.getLastUpdated() != null) {
                            return ParserUtils.relativeTime(model.getLastUpdated(), "MMMM d, yyyy h:mm a", false);
                        } else {
                            return "";
                        }
                    }

                });

        cameraCellList = new CellList<CameraItem>(new CameraCell<CameraItem>() {

            @Override
            public String getUrl(CameraItem model) {
                return model.getImageUrl();
            }

            @Override
            public boolean canBeSelected(CameraItem model) {
                return true;
            }
        });

        initWidget(uiBinder.createAndBindUi(this));

        accessibilityPrepare();

        if (MGWT.getOsDetection().isAndroid()) {
            leftFlexSpacer.setVisible(false);
            cameraScrollPanel.setBounce(false);
        }
    }

    /**
     * ScrollPanel doesn't allow scrolling to the bottom if it contains a CellList with images.
     * 
     * See: https://code.google.com/p/mgwt/issues/detail?id=276
     * 
     * ScrollPanel.refresh() must be explicitly called after the images are loaded.
     * Since the onload event of images is not bubbling up, the LoadHandler can't be attached
     * to the CellList. Instead, the onload event needs to be captured at the <img>, and directly
     * trigger the ScrollPanel.refresh() from there.
     */
    private native void handleOnLoad() /*-{
                                       $wnd.refreshPanel = @gov.wa.wsdot.mobile.client.activities.ferries.schedules.departures.FerriesRouteDeparturesViewGwtImpl::refreshPanel();
                                       }-*/;

    public static void refreshPanel() {
        cameraScrollPanel.refresh();
    }

    @UiHandler("tabPanel")
    protected void onTabSelected(SelectionEvent<Integer> event) {
        if (presenter != null) {
            int index = event.getSelectedItem();
            presenter.onTabSelected(index);
        }
    }

    @UiHandler("timesTab")
    protected void onTimesTabPressed(TapEvent event) {
        if (presenter != null) {
            accessibilityShowTimes();
        }
    }

    @UiHandler("camerasTab")
    protected void onCamerasTabPressed(TapEvent event) {
        if (presenter != null) {
            accessibilityShowCameras();

        }
    }

    @UiHandler("backButton")
    protected void onBackButtonPressed(TapEvent event) {
        if (presenter != null) {
            presenter.onBackButtonPressed();
        }
    }

    @UiHandler("daysOfWeek")
    protected void onChange(ChangeEvent event) {
        if (presenter != null) {
            MListBox source = (MListBox) event.getSource();
            presenter.onDayOfWeekSelected(source.getSelectedIndex());
        }
    }

    @UiHandler("cameraCellList")
    protected void onCameraCellSelected(CellSelectedEvent event) {
        if (presenter != null) {
            int index = event.getIndex();
            presenter.onCameraSelected(index);
        }
    }

    @Override
    public void setPresenter(Presenter presenter) {
        this.presenter = presenter;
    }

    @Override
    public void setTitle(String title) {
        this.title.setHTML(title);
    }

    @Override
    public void render(List<FerriesScheduleTimesItem> departureTimesList) {
        cellList.render(departureTimesList);
    }

    @Override
    public void renderCameras(List<CameraItem> cameraList) {
        cameraCellList.render(cameraList);
    }

    @Override
    public void showProgressIndicator() {
        progressIndicator.setVisible(true);
    }

    @Override
    public void hideProgressIndicator() {
        progressIndicator.setVisible(false);
    }

    @Override
    public void refresh() {
        pullToRefresh.refresh();
    }

    @Override
    public void refreshCameras() {
        cameraScrollPanel.refresh();
    }

    @Override
    public int getDayOfWeekSelected() {
        return daysOfWeek.getSelectedIndex();
    }

    @Override
    public void setDayOfWeekSelected(int index) {
        daysOfWeek.setSelectedIndex(index);
    }

    @Override
    public void renderDaysOfWeek(List<String> days) {
        daysOfWeek.clear();

        for (String day : days) {
            daysOfWeek.addItem(dayOfWeekFormat.format(new Date(Long.parseLong(day))));
        }
    }

    @Override
    public void setHeaderPullHandler(Pullhandler pullHandler) {
        pullToRefresh.setHeaderPullHandler(pullHandler);
    }

    @Override
    public PullArrowWidget getPullHeader() {
        return pullArrowHeader;
    }

    @Override
    public HasRefresh getPullPanel() {
        return pullToRefresh;
    }

    @Override
    public void setCameraSelected(int lastIndex, boolean b) {
        cameraCellList.setSelectedIndex(lastIndex, b);
    }

    @Override
    public void removeTab(int tabIndex) {
        this.tabPanel.tabBar.remove(tabIndex);
        this.tabPanel.tabContainer.container.remove(tabIndex);
        this.tabPanel.tabContainer.refresh();
    }

    @Override
    public int getTabCount() {
        return this.tabPanel.tabContainer.container.getWidgetCount();
    }

    private void accessibilityShowTimes() {
        Roles.getMainRole().setAriaHiddenState(times.getElement(), false);
        Roles.getMainRole().setAriaHiddenState(cameras.getElement(), true);
        Roles.getTabRole().setAriaSelectedState(timesTab.getElement(), SelectedValue.TRUE);
        Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.FALSE);
    }

    private void accessibilityShowCameras() {
        Roles.getMainRole().setAriaHiddenState(times.getElement(), true);
        Roles.getMainRole().setAriaHiddenState(cameras.getElement(), false);
        Roles.getTabRole().setAriaSelectedState(timesTab.getElement(), SelectedValue.FALSE);
        Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.TRUE);
    }

    private void accessibilityPrepare() {

        // Add ARIA roles for accessibility
        Roles.getButtonRole().set(backButton.getElement());
        Roles.getButtonRole().setAriaLabelProperty(backButton.getElement(), "back");

        Roles.getHeadingRole().set(heading.getElement());

        Roles.getMenuRole().set(daysOfWeek.getElement());
        Roles.getMenuRole().setAriaLabelProperty(daysOfWeek.getElement(), "select a departing day");
        Roles.getMenuRole().setTabindexExtraAttribute(daysOfWeek.getElement(), 0);

        Roles.getMainRole().set(times.getElement());
        Roles.getMainRole().set(cameras.getElement());

        Roles.getTabRole().set(timesTab.getElement());
        Roles.getTabRole().setAriaSelectedState(timesTab.getElement(), SelectedValue.TRUE);
        Roles.getTabRole().setAriaLabelProperty(timesTab.getElement(), "times");

        Roles.getTabRole().set(camerasTab.getElement());
        Roles.getTabRole().setAriaSelectedState(camerasTab.getElement(), SelectedValue.FALSE);
        Roles.getTabRole().setAriaLabelProperty(camerasTab.getElement(), "cameras");

        Roles.getProgressbarRole().set(progressIndicator.getElement());
        Roles.getProgressbarRole().setAriaLabelProperty(progressIndicator.getElement(), "loading indicator");

        // TODO Hide pull down until we can figure out how to get VoiceOver to work with it
        Roles.getButtonRole().setAriaHiddenState(pullArrowHeader.getElement(), true);

        accessibilityShowTimes();
    }
}