/* * 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(); } }