diff --git a/xwords4/android/XWords4/AndroidManifest.xml b/xwords4/android/XWords4/AndroidManifest.xml
index 58e8d8dab..3a40bc0f1 100644
--- a/xwords4/android/XWords4/AndroidManifest.xml
+++ b/xwords4/android/XWords4/AndroidManifest.xml
@@ -97,7 +97,7 @@
+
+
+
+
+
+
+
+
diff --git a/xwords4/android/XWords4/archive/R.java b/xwords4/android/XWords4/archive/R.java
index 4b05e876d..ecabfd403 100644
--- a/xwords4/android/XWords4/archive/R.java
+++ b/xwords4/android/XWords4/archive/R.java
@@ -321,11 +321,11 @@ public final class R {
public static final class string {
/** copyright info
*/
- public static final int about_copyright=0x7f0601e3;
+ public static final int about_copyright=0x7f0601e6;
/** Another paragraph giving credit for work done other than by
Eric House and translators
*/
- public static final int about_credits=0x7f0601e6;
+ public static final int about_credits=0x7f0601e9;
/**
###########################################################
# :Dialogs:
@@ -336,22 +336,22 @@ public final class R {
String giving version info, which is substituted in.
*/
- public static final int about_vers_fmt=0x7f0601e2;
+ public static final int about_vers_fmt=0x7f0601e5;
/** Another paragraph in the about dialog
*/
- public static final int about_web=0x7f0601e4;
- public static final int add_done_fmt=0x7f06029c;
- public static final int add_to_study_fmt=0x7f060294;
+ public static final int about_web=0x7f0601e7;
+ public static final int add_done_fmt=0x7f06029f;
+ public static final int add_to_study_fmt=0x7f060297;
/** Debugging stuff. No point in localizing it.
*/
- public static final int advanced=0x7f060074;
+ public static final int advanced=0x7f060075;
/** This button takes you to the normal Game Configure screen
*/
- public static final int advanced_config=0x7f06019e;
- public static final int advanced_summary=0x7f060075;
+ public static final int advanced_config=0x7f0601a1;
+ public static final int advanced_summary=0x7f060076;
/**
*/
- public static final int alert_empty_dict_fmt=0x7f060200;
+ public static final int alert_empty_dict_fmt=0x7f060203;
/** The name of the app. Not localized!
*/
public static final int app_name=0x7f060001;
@@ -359,22 +359,22 @@ public final class R {
/** the background color of the area outside the board,
e.g. between entries in the scoreboard
*/
- public static final int background=0x7f06015b;
+ public static final int background=0x7f06015e;
/** Appended to the above in the phonies_warn case. User may
ignore the warning
*/
- public static final int badwords_accept=0x7f0600fd;
+ public static final int badwords_accept=0x7f060100;
/** Appended to the above in the phonies_disallow case. User has
lost his turn.
*/
- public static final int badwords_lost=0x7f0600fe;
+ public static final int badwords_lost=0x7f060101;
/** title of the dialog in which the above is posted.
*/
- public static final int badwords_title=0x7f0600ff;
- public static final int blue=0x7f06015f;
+ public static final int badwords_title=0x7f060102;
+ public static final int blue=0x7f060162;
/** board menu for small devices only
*/
- public static final int board_menu_dict=0x7f06023d;
+ public static final int board_menu_dict=0x7f060240;
/**
############################################################
# :Menus:
@@ -384,150 +384,150 @@ public final class R {
This menu commits the current move as it's been laid out on
the board
*/
- public static final int board_menu_done=0x7f060100;
+ public static final int board_menu_done=0x7f060103;
/** Brings up "About Crosswords" dialog
*/
- public static final int board_menu_file_about=0x7f06009b;
+ public static final int board_menu_file_about=0x7f06009e;
/**
*/
- public static final int board_menu_file_email=0x7f060205;
+ public static final int board_menu_file_email=0x7f060208;
/** menu on Game submenu: brings up dialog listing all tiles in
the language of the game along with how many of each there
are and how many points each is worth. This display is the
same throughout the game.
*/
- public static final int board_menu_game_counts=0x7f060106;
+ public static final int board_menu_game_counts=0x7f060109;
/** Brings up explanation of the game's final score. If the game
is not yet over, gives you a choice whether to end it now,
and if you decline does nothing.
*/
- public static final int board_menu_game_final=0x7f060109;
+ public static final int board_menu_game_final=0x7f06010c;
/** Brings up listing of all moves played so far this game.
*/
- public static final int board_menu_game_history=0x7f060108;
+ public static final int board_menu_game_history=0x7f06010b;
/** menu on Game submenu: brings up dialog listing all tiles not
yet played and not in the rack of the player whose rack is
visible (whose turn it is, generally). This display will
change as tiles are used and depending on what player is
asking.
*/
- public static final int board_menu_game_left=0x7f060107;
- public static final int board_menu_game_netstats=0x7f06007d;
+ public static final int board_menu_game_left=0x7f06010a;
+ public static final int board_menu_game_netstats=0x7f060080;
/** Meaningful only for networked games, this menu causes all
messages that have not yet been acknowledged by a remote
device in the game to be resent. Eventually I hope to be
able to remove this from non-debug versions of the game
because users should not have to do do this EVER.
*/
- public static final int board_menu_game_resend=0x7f06010b;
- public static final int board_menu_game_resign=0x7f06010a;
- public static final int board_menu_invite=0x7f06028a;
+ public static final int board_menu_game_resend=0x7f06010e;
+ public static final int board_menu_game_resign=0x7f06010d;
+ public static final int board_menu_invite=0x7f06028d;
/**
*/
- public static final int board_menu_pass=0x7f0601f2;
+ public static final int board_menu_pass=0x7f0601f5;
/** This menu begins an exchange of tiles: puts the board into
trade mode.
*/
- public static final int board_menu_trade=0x7f060101;
+ public static final int board_menu_trade=0x7f060104;
/** hide and shows the tray. On devices where there is enough
room for the full board and tray to be shown then hiding the
tray just "turns it over", i.e. shows tiles with '?'
characters.
*/
- public static final int board_menu_tray_hide=0x7f060102;
- public static final int board_menu_tray_show=0x7f060103;
+ public static final int board_menu_tray_hide=0x7f060105;
+ public static final int board_menu_tray_show=0x7f060106;
/**
*/
- public static final int board_menu_undo_current=0x7f060237;
+ public static final int board_menu_undo_current=0x7f06023a;
/** Undos the last *committed* turn. Note that this is different
from the undo button that undoes or redoes an in-progress
not-yet-committed turn. This is disabled for networked
games.
*/
- public static final int board_menu_undo_last=0x7f060104;
+ public static final int board_menu_undo_last=0x7f060107;
/** preference for board size (15x15, 13x13 etc.)
*/
- public static final int board_size=0x7f060140;
+ public static final int board_size=0x7f060143;
/** Title of submenu
*/
- public static final int board_submenu_game=0x7f060105;
+ public static final int board_submenu_game=0x7f060108;
/** (color for) double-letter bonus squares on the board
*/
- public static final int bonus_l2x=0x7f060154;
+ public static final int bonus_l2x=0x7f060157;
/** Bonus value hint that's displayed in gray text in the colored
bonus square. Double-letter
*/
- public static final int bonus_l2x_summary=0x7f0600eb;
+ public static final int bonus_l2x_summary=0x7f0600ee;
/** (color for) triple-letter bonus squares on the board
*/
- public static final int bonus_l3x=0x7f060155;
+ public static final int bonus_l3x=0x7f060158;
/** Bonus value hint that's displayed in gray text in the colored
bonus square. Triple-letter
*/
- public static final int bonus_l3x_summary=0x7f0600ed;
+ public static final int bonus_l3x_summary=0x7f0600f0;
/** (color for) double-word squares on the board
*/
- public static final int bonus_w2x=0x7f060156;
+ public static final int bonus_w2x=0x7f060159;
/** Bonus value hint that's displayed in gray text in the colored
bonus square. Double-word
*/
- public static final int bonus_w2x_summary=0x7f0600ec;
+ public static final int bonus_w2x_summary=0x7f0600ef;
/** (color for) triple-word squares on the board
*/
- public static final int bonus_w3x=0x7f060157;
+ public static final int bonus_w3x=0x7f06015a;
/** Bonus value hint that's displayed in gray text in the colored
bonus square. Triple-word
*/
- public static final int bonus_w3x_summary=0x7f0600ee;
+ public static final int bonus_w3x_summary=0x7f0600f1;
/**
*/
- public static final int bt_bad_proto_fmt=0x7f060214;
+ public static final int bt_bad_proto_fmt=0x7f060217;
/** EXPLAIN ME
*/
- public static final int bt_disabled=0x7f060184;
+ public static final int bt_disabled=0x7f060187;
/**
*/
- public static final int bt_fail_fmt=0x7f060219;
+ public static final int bt_fail_fmt=0x7f06021c;
/** EXPLAIN ME
*/
- public static final int bt_networked_desc=0x7f060185;
+ public static final int bt_networked_desc=0x7f060188;
/**
*/
- public static final int bt_pick_clear_button=0x7f06020b;
+ public static final int bt_pick_clear_button=0x7f06020e;
/**
*/
- public static final int bt_pick_rescan_button=0x7f06020a;
+ public static final int bt_pick_rescan_button=0x7f06020d;
/**
*/
- public static final int bt_resend_fmt=0x7f060218;
+ public static final int bt_resend_fmt=0x7f06021b;
/**
*/
- public static final int btname_label=0x7f06021b;
+ public static final int btname_label=0x7f06021e;
/** text of button for adding new player to game
*/
- public static final int button_add_player=0x7f0600c1;
- public static final int button_cancel=0x7f0601bb;
+ public static final int button_add_player=0x7f0600c4;
+ public static final int button_cancel=0x7f0601be;
/** The only button available when the above message is displayed
*/
- public static final int button_close_game=0x7f0601af;
- public static final int button_decline=0x7f060266;
+ public static final int button_close_game=0x7f0601b2;
+ public static final int button_decline=0x7f060269;
/**
*/
- public static final int button_default_both=0x7f0600b6;
+ public static final int button_default_both=0x7f0600b9;
/** These three strings are the text for three buttons giving
choices in respose to the dialog launched in response to the
dicts_item_select menu (see dicts_item_select above.)
*/
- public static final int button_default_human=0x7f0600b4;
+ public static final int button_default_human=0x7f0600b7;
/**
*/
- public static final int button_default_robot=0x7f0600b5;
+ public static final int button_default_robot=0x7f0600b8;
/** When you select the gamel_menu_delete_all menuitem or
list_item_delete contextual menu, you are asked to confirm.
This is the text of the first button ("Cancel" is the
second).
*/
- public static final int button_delete=0x7f060094;
+ public static final int button_delete=0x7f060097;
/** Used for button in dialog put up with the relay says the
remote partner of this game has deleted it. Ok, meaning
"don't discard", is the other option. This same button is
@@ -535,36 +535,36 @@ public final class R {
the other option, for discarding changes that would otherwise
reset a game.
*/
- public static final int button_discard=0x7f0601bf;
+ public static final int button_discard=0x7f0601c2;
/**
*/
- public static final int button_done=0x7f0601ef;
+ public static final int button_done=0x7f0601f2;
/**
*/
- public static final int button_done_fmt=0x7f0601f0;
+ public static final int button_done_fmt=0x7f0601f3;
/** Text of button displayed when downloading is an option
*/
- public static final int button_download=0x7f0601b2;
- public static final int button_go_settings=0x7f06028c;
- public static final int button_html=0x7f06018a;
+ public static final int button_download=0x7f0601b5;
+ public static final int button_go_settings=0x7f06028f;
+ public static final int button_html=0x7f06018d;
/**
*/
- public static final int button_invite=0x7f06021a;
+ public static final int button_invite=0x7f06021d;
/** text of button to juggle (randomly rearrange order of) players
*/
- public static final int button_juggle_players=0x7f0600c2;
+ public static final int button_juggle_players=0x7f0600c5;
/** New strings that need to be documented and found a home
above.
*/
- public static final int button_lookup=0x7f0601e9;
+ public static final int button_lookup=0x7f0601ec;
/**
*/
- public static final int button_lookup_fmt=0x7f0601eb;
- public static final int button_lookup_study=0x7f0601ea;
- public static final int button_lookup_study_fmt=0x7f0601ec;
+ public static final int button_lookup_fmt=0x7f0601ee;
+ public static final int button_lookup_study=0x7f0601ed;
+ public static final int button_lookup_study_fmt=0x7f0601ef;
/**
*/
- public static final int button_move=0x7f0601f4;
+ public static final int button_move=0x7f0601f7;
/**
############################################################
# :Screens:
@@ -575,16 +575,16 @@ public final class R {
Text of menuitem in main games-list screen's menu
*/
- public static final int button_new_game=0x7f060089;
+ public static final int button_new_game=0x7f06008c;
/** Games list screen menuitem text for creating a new group
*/
- public static final int button_new_group=0x7f06008a;
- public static final int button_nfc=0x7f06018b;
- public static final int button_no=0x7f0601bd;
+ public static final int button_new_group=0x7f06008d;
+ public static final int button_nfc=0x7f06018e;
+ public static final int button_no=0x7f0601c0;
/** Text for button in new-user-info dialog with title just
above.
*/
- public static final int button_notagain=0x7f0601cb;
+ public static final int button_notagain=0x7f0601ce;
/**
############################################################
# Dialogs
@@ -593,23 +593,23 @@ public final class R {
Text for buttons at the bottom of dialogs. These first are
in many places.
*/
- public static final int button_ok=0x7f0601ba;
- public static final int button_reconnect=0x7f06027c;
+ public static final int button_ok=0x7f0601bd;
+ public static final int button_reconnect=0x7f06027f;
/** Button shown in game over dialog triggering creation of new
game with the same players and parameters as the one that
just ended.
*/
- public static final int button_rematch=0x7f06027b;
+ public static final int button_rematch=0x7f06027e;
/** When you select the list_item_reset contextual menu, you are
asked to confirm. This is the text of the first button
("Cancel" is the second).
*/
- public static final int button_reset=0x7f060095;
+ public static final int button_reset=0x7f060098;
/** Used for a button when informing user that his attempt to
connect to the relay failed because the room named does not
exist. (I believe this no longer occurs.)
*/
- public static final int button_retry=0x7f0601c0;
+ public static final int button_retry=0x7f0601c3;
/**
############################################################
# :Menus:
@@ -622,30 +622,30 @@ public final class R {
text of first menu item. Will revert all preferences to
their default/original values
*/
- public static final int button_revert_all=0x7f060129;
+ public static final int button_revert_all=0x7f06012c;
/** Second menu item. Reverts only the colors to their
default/original values
*/
- public static final int button_revert_colors=0x7f06012b;
+ public static final int button_revert_colors=0x7f06012e;
/** Used in Game config dialog to confirm saving changes that reset a game
*/
- public static final int button_save=0x7f0601be;
+ public static final int button_save=0x7f0601c1;
/**
*/
- public static final int button_search=0x7f0601f5;
+ public static final int button_search=0x7f0601f8;
/**
*/
- public static final int button_sms_add=0x7f060228;
+ public static final int button_sms_add=0x7f06022b;
/** Text of button allowing user to choose to open with a
different (but same-language wordlist)
*/
- public static final int button_substdict=0x7f0601b3;
+ public static final int button_substdict=0x7f0601b6;
/** When an invitation is sent, the user gets to choose between
plaintext and html formatting. These two strings are shown in the
two buttons in the dialog.
*/
- public static final int button_text=0x7f060189;
- public static final int button_trade_cancel=0x7f0600e9;
+ public static final int button_text=0x7f06018c;
+ public static final int button_trade_cancel=0x7f0600ec;
/**
############################################################
# :Screens:
@@ -659,21 +659,21 @@ public final class R {
Buttons shown at bottom when board is in exchange mode,
i.e. after user has picked board_menu_trade menu item
*/
- public static final int button_trade_commit=0x7f0600e8;
- public static final int button_yes=0x7f0601bc;
- public static final int cannot_delete_default_group_fmt=0x7f060278;
- public static final int change_group=0x7f06027f;
+ public static final int button_trade_commit=0x7f0600eb;
+ public static final int button_yes=0x7f0601bf;
+ public static final int cannot_delete_default_group_fmt=0x7f06027b;
+ public static final int change_group=0x7f060282;
/** text of button in About Crosswords dialog summoning above
dialog
*/
- public static final int changes_button=0x7f0601e8;
+ public static final int changes_button=0x7f0601eb;
/** text of dialog showing the set of changes made since the last
release
*/
- public static final int changes_title=0x7f0601e7;
+ public static final int changes_title=0x7f0601ea;
/** Prefix for local messages
*/
- public static final int chat_local_id=0x7f060194;
+ public static final int chat_local_id=0x7f060197;
/**
############################################################
# :Menus:
@@ -685,14 +685,14 @@ public final class R {
other devices in the game: no "clear chat" message is
transmitted.
*/
- public static final int chat_menu_clear=0x7f060197;
+ public static final int chat_menu_clear=0x7f06019a;
/** Prefix for remote messages
*/
- public static final int chat_other_id=0x7f060195;
+ public static final int chat_other_id=0x7f060198;
/** Text on the button that causes the contents of the
message-composition field to be sent.
*/
- public static final int chat_send=0x7f060196;
+ public static final int chat_send=0x7f060199;
/** ############################################################
# :Screens:
# Chat screen
@@ -707,81 +707,81 @@ public final class R {
title of the chat screen. The name of the current game is
substituted for %1$s.
*/
- public static final int chat_title_fmt=0x7f060193;
- public static final int checkupdates_none_found=0x7f060255;
+ public static final int chat_title_fmt=0x7f060196;
+ public static final int checkupdates_none_found=0x7f060258;
/** color of the "crosshairs", lines drawn vertically and
horizontally through the square the user is currently
touching in order to guide the fat-fingered (most of us) in
operations that require accurately selecting a single square
on the board.
*/
- public static final int clr_crosshairs=0x7f060158;
+ public static final int clr_crosshairs=0x7f06015b;
/**
*/
- public static final int confirm_clear=0x7f06022e;
+ public static final int confirm_clear=0x7f060231;
/** text of confirmation dialog posted when the delete 'X' button
beside the listing of a wordlist is tapped. The name of the
wordlist is substituted for %1$s. Sometimes one of the two
strings below is appended.
*/
- public static final int confirm_delete_dict_fmt=0x7f0600b0;
+ public static final int confirm_delete_dict_fmt=0x7f0600b3;
/** Additional text appended to text confirm_delete_dictf in the
wordlist delete confiration dialog in the case where the
wordlist to be deleted is NOT the last in its language. The
name of the language is substituted for %1$s.
*/
- public static final int confirm_deletemore_dict_fmt=0x7f0600b2;
+ public static final int confirm_deletemore_dict_fmt=0x7f0600b5;
/** Additional text appended to text confirm_delete_dictf in the
wordlist delete confiration dialog in the case where the
wordlist to be deleted is the last in its language. The name
of the language is substituted for %1$s.
*/
- public static final int confirm_deleteonly_dict_fmt=0x7f0600b1;
+ public static final int confirm_deleteonly_dict_fmt=0x7f0600b4;
/** Text of confirmation dialog posted when list_item_reset menu
is selected
*/
- public static final int confirm_reset_fmt=0x7f0600a8;
+ public static final int confirm_reset_fmt=0x7f0600ab;
/** Text of confirmation dialog for above
*/
- public static final int confirm_revert_all=0x7f06012c;
+ public static final int confirm_revert_all=0x7f06012f;
/** Text of confirmation dialog for above
*/
- public static final int confirm_revert_colors=0x7f06012a;
+ public static final int confirm_revert_colors=0x7f06012d;
/** title of confirmation dialog put up when user has unlocked an
in-progress game and wants to save changes (has hit the
play button).
*/
- public static final int confirm_save=0x7f0600db;
+ public static final int confirm_save=0x7f0600de;
/** title of above confirmation dialog
*/
- public static final int confirm_save_title=0x7f0600dc;
- public static final int confirm_seldeletes_fmt=0x7f0600a7;
+ public static final int confirm_save_title=0x7f0600df;
+ public static final int confirm_seldeletes_fmt=0x7f0600aa;
/**
*/
- public static final int confirm_sms_expl=0x7f06024e;
+ public static final int confirm_sms_expl=0x7f060251;
/**
*/
- public static final int confirm_sms_leave=0x7f060250;
+ public static final int confirm_sms_leave=0x7f060253;
/**
*/
- public static final int confirm_sms_prompt=0x7f06024f;
+ public static final int confirm_sms_prompt=0x7f060252;
/**
*/
- public static final int confirm_sms_title=0x7f06024d;
+ public static final int confirm_sms_title=0x7f060250;
/**
*/
- public static final int confirm_sms_unlimited=0x7f060251;
+ public static final int confirm_sms_unlimited=0x7f060254;
/**
*/
- public static final int confirm_sms_willpay=0x7f060252;
- public static final int confirm_studylist_clear_fmt=0x7f06029a;
+ public static final int confirm_sms_willpay=0x7f060255;
+ public static final int confirm_studylist_clear_fmt=0x7f06029d;
/** text of dialog shown when the menu item board_menu_undo_last
is chosen.
*/
- public static final int confirm_undo_last=0x7f060127;
- public static final int connect_daily=0x7f060177;
- public static final int connect_fifteen_mins=0x7f060173;
- public static final int connect_five_mins=0x7f060172;
+ public static final int confirm_undo_last=0x7f06012a;
+ public static final int connect_daily=0x7f06017a;
+ public static final int connect_fifteen_mins=0x7f060176;
+ public static final int connect_five_mins=0x7f060175;
/**
############################################################
# :Screens:
@@ -794,80 +794,81 @@ public final class R {
new moves available and fetches them. This controls how
frequently that check is done.
*/
- public static final int connect_frequency=0x7f060170;
+ public static final int connect_frequency=0x7f060173;
/** text of separator marking out the connection area of the dialog
*/
- public static final int connect_label_relay=0x7f0600c5;
+ public static final int connect_label_relay=0x7f0600c8;
/**
*/
- public static final int connect_label_sms=0x7f060230;
+ public static final int connect_label_sms=0x7f060233;
/** These are the possible values for the connect_frequency
setting presented as a drop-down list.
*/
- public static final int connect_never=0x7f060171;
- public static final int connect_one_hour=0x7f060175;
- public static final int connect_six_hours=0x7f060176;
- public static final int connect_thirty_mins=0x7f060174;
+ public static final int connect_never=0x7f060174;
+ public static final int connect_one_hour=0x7f060178;
+ public static final int connect_six_hours=0x7f060179;
+ public static final int connect_thirty_mins=0x7f060177;
/**
*/
- public static final int connstat_bt=0x7f06024a;
+ public static final int connstat_bt=0x7f06024d;
/**
*/
- public static final int connstat_lastother_succ_fmt=0x7f060244;
+ public static final int connstat_lastother_succ_fmt=0x7f060247;
/**
*/
- public static final int connstat_lastother_unsucc_fmt=0x7f060245;
+ public static final int connstat_lastother_unsucc_fmt=0x7f060248;
/**
*/
- public static final int connstat_lastreceipt_fmt=0x7f060246;
+ public static final int connstat_lastreceipt_fmt=0x7f060249;
/**
*/
- public static final int connstat_lastsend_fmt=0x7f060243;
+ public static final int connstat_lastsend_fmt=0x7f060246;
/**
*/
- public static final int connstat_net_fmt=0x7f060240;
+ public static final int connstat_net_fmt=0x7f060243;
/**
*/
- public static final int connstat_nonet=0x7f06023f;
+ public static final int connstat_nonet=0x7f060242;
/**
*/
- public static final int connstat_noreceipt=0x7f060247;
+ public static final int connstat_noreceipt=0x7f06024a;
/**
*/
- public static final int connstat_relay=0x7f060248;
+ public static final int connstat_relay=0x7f06024b;
/**
*/
- public static final int connstat_sms=0x7f060249;
+ public static final int connstat_sms=0x7f06024c;
/**
*/
- public static final int connstat_succ=0x7f060241;
+ public static final int connstat_succ=0x7f060244;
/**
*/
- public static final int connstat_unsucc=0x7f060242;
+ public static final int connstat_unsucc=0x7f060245;
/** title of dialog brought up in response to the
board_menu_game_counts menu. The dialog lists all tiles in
the language being used for the game together with how many
of each there are are and how many points each is worth.
*/
- public static final int counts_values_title=0x7f0601c5;
- public static final int cur_menu_marker_fmt=0x7f060289;
+ public static final int counts_values_title=0x7f0601c8;
+ public static final int cur_menu_marker_fmt=0x7f06028c;
/**
*/
- public static final int cur_tiles_fmt=0x7f0601f9;
- public static final int debug_features=0x7f06007b;
- public static final int debug_features_summary=0x7f06007c;
+ public static final int cur_tiles_fmt=0x7f0601fc;
+ public static final int data_gsm_only=0x7f0602ae;
+ public static final int debug_features=0x7f06007c;
+ public static final int debug_features_summary=0x7f06007d;
/** dictionary used by default for human players when creating
new game
*/
- public static final int default_dict=0x7f060138;
+ public static final int default_dict=0x7f06013b;
/** other
*/
- public static final int default_host=0x7f060069;
- public static final int default_loc=0x7f060269;
- public static final int default_loc_summary=0x7f06026a;
+ public static final int default_host=0x7f06006a;
+ public static final int default_loc=0x7f06026c;
+ public static final int default_loc_summary=0x7f06026d;
/** Welcome dialog text
*/
- public static final int default_name_message=0x7f0601e1;
+ public static final int default_name_message=0x7f0601e4;
/**
###########################################################
# :Dialogs:
@@ -882,41 +883,41 @@ public final class R {
Welcome dialog title
*/
- public static final int default_name_title=0x7f0601e0;
+ public static final int default_name_title=0x7f0601e3;
/** default new-game setting for handline phonies (words not
found in the word list)
*/
- public static final int default_phonies=0x7f06013e;
+ public static final int default_phonies=0x7f060141;
/** dictionary used by default for robot players when creating
new game
*/
- public static final int default_robodict=0x7f060139;
- public static final int default_update_url=0x7f060073;
+ public static final int default_robodict=0x7f06013c;
+ public static final int default_update_url=0x7f060074;
/**
*/
- public static final int dft_name_fmt=0x7f06021f;
+ public static final int dft_name_fmt=0x7f060222;
/**
*/
- public static final int dft_sms_name_fmt=0x7f060224;
+ public static final int dft_sms_name_fmt=0x7f060227;
/**
*/
- public static final int dict_browse_nowords_fmt=0x7f0601fd;
+ public static final int dict_browse_nowords_fmt=0x7f060200;
/**
*/
- public static final int dict_browse_title1_fmt=0x7f0601fc;
+ public static final int dict_browse_title1_fmt=0x7f0601ff;
/**
*/
- public static final int dict_browse_title_fmt=0x7f0601fb;
- public static final int dict_host=0x7f060078;
+ public static final int dict_browse_title_fmt=0x7f0601fe;
+ public static final int dict_host=0x7f060079;
/** label for dropdown by which wordlist is chosen that this
player will use. The language the game will use (which
constrains the choice of wordlists) is substituted in for
"%1$s".
*/
- public static final int dict_lang_label_fmt=0x7f0600e4;
+ public static final int dict_lang_label_fmt=0x7f0600e7;
/** string name="invite_mime">text/plainDisplay snapshots of games
*/
- public static final int dropped_dupe=0x7f060288;
+ public static final int dropped_dupe=0x7f06028b;
/** Shown in the main screen when you launch Crosswords from an
invitation (received in email or messaging app, say) and
there's already a game running that matches that invitation.
@@ -962,61 +963,61 @@ public final class R {
the same room name over and over so they'll get this warning
and it's harmless to ignore it.
*/
- public static final int dup_game_query_fmt=0x7f0601c1;
+ public static final int dup_game_query_fmt=0x7f0601c4;
/**
*/
- public static final int email_author_chooser=0x7f060207;
+ public static final int email_author_chooser=0x7f06020a;
/** Nor is my email address
*/
- public static final int email_author_email=0x7f060067;
+ public static final int email_author_email=0x7f060068;
/**
*/
- public static final int email_author_subject=0x7f060206;
+ public static final int email_author_subject=0x7f060209;
/**
*/
- public static final int email_body_rev_fmt=0x7f060208;
+ public static final int email_body_rev_fmt=0x7f06020b;
/** color of empty squares on the board (that are not bonus squares)
*/
- public static final int empty=0x7f06015a;
+ public static final int empty=0x7f06015d;
/**
*/
- public static final int empty_sms_inviter=0x7f06022c;
- public static final int enable_dupes_summary=0x7f060087;
- public static final int enable_dupes_title=0x7f060085;
- public static final int enable_nfc=0x7f06028b;
+ public static final int empty_sms_inviter=0x7f06022f;
+ public static final int enable_dupes_summary=0x7f06008a;
+ public static final int enable_dupes_title=0x7f060088;
+ public static final int enable_nfc=0x7f06028e;
/**
*/
- public static final int enable_sms=0x7f06024b;
+ public static final int enable_sms=0x7f06024e;
/**
*/
- public static final int enable_sms_summary=0x7f06024c;
- public static final int entering_trade=0x7f0600ea;
- public static final int expl_update_url=0x7f060072;
+ public static final int enable_sms_summary=0x7f06024f;
+ public static final int entering_trade=0x7f0600ed;
+ public static final int expl_update_url=0x7f060073;
/** These four strings are displayed as step-by-step
instructions
*/
- public static final int explain_b1=0x7f060199;
- public static final int explain_b2=0x7f06019a;
- public static final int explain_b3=0x7f06019b;
- public static final int explain_b4=0x7f06019c;
+ public static final int explain_b1=0x7f06019c;
+ public static final int explain_b2=0x7f06019d;
+ public static final int explain_b3=0x7f06019e;
+ public static final int explain_b4=0x7f06019f;
/** if this preference is checked, a dialog will be posted every
time a robot makes a move or a move is received from a remote
player.
*/
- public static final int explain_robot=0x7f060162;
+ public static final int explain_robot=0x7f060165;
/** explanation of the above
*/
- public static final int explain_robot_summary=0x7f060163;
+ public static final int explain_robot_summary=0x7f060166;
/** title of dialog brought up in response to the
board_menu_game_final menu. The dialog displays the final
score and an accounting of it (including subtractions for
running out the game timer if there is one.)
*/
- public static final int finalscores_title=0x7f0601c7;
+ public static final int finalscores_title=0x7f0601ca;
/** Explanatory text appears in the dialog
*/
- public static final int force_expl=0x7f0600df;
+ public static final int force_expl=0x7f0600e2;
/**
############################################################
# :Dialogs:
@@ -1031,22 +1032,22 @@ public final class R {
title of this dialog
*/
- public static final int force_title=0x7f0600de;
+ public static final int force_title=0x7f0600e1;
/** Displayed as a "Toast" (mini window briefly displayed) if you
didn't choose a remote player and I was forced to pick one. I
always pick the first.
*/
- public static final int forced_consistent=0x7f0600e0;
+ public static final int forced_consistent=0x7f0600e3;
/** Games that have ended are listed with this string
*/
- public static final int gameOver=0x7f060092;
+ public static final int gameOver=0x7f060095;
/**
*/
- public static final int game_btname_title=0x7f06021d;
+ public static final int game_btname_title=0x7f060220;
/** used to create default names of games (when user has not
named them.)
*/
- public static final int game_fmt=0x7f0601b6;
+ public static final int game_fmt=0x7f0601b9;
/**
############################################################
# :Menus:
@@ -1056,46 +1057,46 @@ public final class R {
title of contextual menu. Name of the selected game is
substituted in.
*/
- public static final int game_item_menu_title_fmt=0x7f06009c;
+ public static final int game_item_menu_title_fmt=0x7f06009f;
/**
*/
- public static final int game_list_tmp=0x7f06023e;
+ public static final int game_list_tmp=0x7f060241;
/** text of checkbox at top of dialog allowing to unlock in-play
game to make changes
*/
- public static final int game_locked=0x7f0600be;
- public static final int game_name_group_title=0x7f060277;
+ public static final int game_locked=0x7f0600c1;
+ public static final int game_name_group_title=0x7f06027a;
/**
*/
- public static final int game_rename_title=0x7f0600a5;
+ public static final int game_rename_title=0x7f0600a8;
/**
*/
- public static final int game_smsname_title=0x7f06021e;
+ public static final int game_smsname_title=0x7f060221;
/** Put nothing in the summary space, so it just reads "Game 2"
*/
- public static final int game_summary_field_empty=0x7f060144;
- public static final int game_summary_field_gameid=0x7f06006f;
+ public static final int game_summary_field_empty=0x7f060147;
+ public static final int game_summary_field_gameid=0x7f060070;
/** Put the language there, so it reads "Game 2 (English)"
*/
- public static final int game_summary_field_language=0x7f060145;
- public static final int game_summary_field_npackets=0x7f060070;
+ public static final int game_summary_field_language=0x7f060148;
+ public static final int game_summary_field_npackets=0x7f060071;
/** List names of opponents (summarized), e.g. "Game 2 (vs Kati)"
*/
- public static final int game_summary_field_opponents=0x7f060146;
- public static final int game_summary_field_rowid=0x7f06006e;
+ public static final int game_summary_field_opponents=0x7f060149;
+ public static final int game_summary_field_rowid=0x7f06006f;
/** List the state of the game, "Game over" or "10 moves made"
etc.
*/
- public static final int game_summary_field_state=0x7f060147;
+ public static final int game_summary_field_state=0x7f06014a;
/** Regardless of the setting of the connect_frequency
preference, checks the relay immediately for any moves for
networked games on this device and posts a notification if
any is downloaded.
*/
- public static final int gamel_menu_checkmoves=0x7f060098;
+ public static final int gamel_menu_checkmoves=0x7f06009b;
/**
*/
- public static final int gamel_menu_checkupdates=0x7f060254;
+ public static final int gamel_menu_checkupdates=0x7f060257;
/**
############################################################
# :Menus:
@@ -1107,25 +1108,25 @@ public final class R {
Brings up the Wordlists (formerly Dictionaries) screen
*/
- public static final int gamel_menu_dicts=0x7f060096;
- public static final int gamel_menu_loaddb=0x7f060084;
- public static final int gamel_menu_storedb=0x7f060083;
- public static final int gamel_menu_study=0x7f060297;
+ public static final int gamel_menu_dicts=0x7f060099;
+ public static final int gamel_menu_loaddb=0x7f060087;
+ public static final int gamel_menu_storedb=0x7f060086;
+ public static final int gamel_menu_study=0x7f06029a;
/**
*/
- public static final int get_sms_number=0x7f06022d;
- public static final int git_rev=0x7f060088;
- public static final int git_rev_title=0x7f06007f;
- public static final int green=0x7f06015e;
- public static final int group_cur_games=0x7f060272;
- public static final int group_name_fmt=0x7f06027a;
- public static final int group_new_games=0x7f060273;
- public static final int groups_confirm_del_fmt=0x7f060274;
- public static final int groups_confirm_del_games_fmt=0x7f060275;
+ public static final int get_sms_number=0x7f060230;
+ public static final int git_rev=0x7f06008b;
+ public static final int git_rev_title=0x7f060082;
+ public static final int green=0x7f060161;
+ public static final int group_cur_games=0x7f060275;
+ public static final int group_name_fmt=0x7f06027d;
+ public static final int group_new_games=0x7f060276;
+ public static final int groups_confirm_del_fmt=0x7f060277;
+ public static final int groups_confirm_del_games_fmt=0x7f060278;
/** Used as the default name for remote players displayed within
the Game configure screen
*/
- public static final int guest_name=0x7f0600da;
+ public static final int guest_name=0x7f0600dd;
/** If this preference is checked the "crosshairs" (vertical and
horzontal lines through the cell your finger is on that help
you tell where the app thinks you're actually tapping) will
@@ -1134,69 +1135,69 @@ public final class R {
large enough screens that they always know where they're
tapping.
*/
- public static final int hide_crosshairs=0x7f06016c;
+ public static final int hide_crosshairs=0x7f06016f;
/** explanation of the above
*/
- public static final int hide_crosshairs_summary=0x7f06016d;
+ public static final int hide_crosshairs_summary=0x7f060170;
/** Checkbox that when set prevents showing a title bar in the
game board window to save space
*/
- public static final int hide_title=0x7f060148;
+ public static final int hide_title=0x7f06014b;
/** clarification of above
*/
- public static final int hide_title_summary=0x7f060149;
+ public static final int hide_title_summary=0x7f06014c;
/** text of checkbox. If this checkbox is checked, buttons will
appear by which the user can get high- and low-scoring moves
generated based on his tray. Can be used to cheat.
*/
- public static final int hints_allowed=0x7f0600cd;
+ public static final int hints_allowed=0x7f0600d0;
/** clarification on hints_allowed, whether new games will
default to having the hint feature enabled(string elsewhere
in this file)
*/
- public static final int hints_allowed_sum=0x7f06013a;
+ public static final int hints_allowed_sum=0x7f06013d;
/** title of dialog brought up in response to the
board_menu_game_history menu. A full history of the game up
to the last turn is displayed, though details about what's in
players' racks is left out if the game is not yet over.
*/
- public static final int history_title=0x7f0601c6;
+ public static final int history_title=0x7f0601c9;
/** Partial text of alert posted when phonies_warn or
phonies_disallow is the current setting and a "phony" is
played. One of the two following strings will be appended
*/
- public static final int ids_badwords_fmt=0x7f0600fc;
+ public static final int ids_badwords_fmt=0x7f0600ff;
/** text of confirmation dialog shown when user chooses the menu
item with text board_menu_game_final and the game is not over
*/
- public static final int ids_endnow=0x7f0601c8;
+ public static final int ids_endnow=0x7f0601cb;
/** Title of generic dialog used to display information
*/
- public static final int info_title=0x7f0601c2;
+ public static final int info_title=0x7f0601c5;
/**
*/
- public static final int inform_dict_diffdict_fmt=0x7f06025e;
+ public static final int inform_dict_diffdict_fmt=0x7f060261;
/**
*/
- public static final int inform_dict_diffversion_fmt=0x7f06025c;
- public static final int inform_dict_download=0x7f06025f;
- public static final int inform_dict_title=0x7f06025d;
+ public static final int inform_dict_diffversion_fmt=0x7f06025f;
+ public static final int inform_dict_download=0x7f060262;
+ public static final int inform_dict_title=0x7f060260;
/** Will new games, on default, randomly rearrange the start
order of players.
*/
- public static final int init_autojuggle=0x7f06013c;
+ public static final int init_autojuggle=0x7f06013f;
/** clarification on above
*/
- public static final int init_autojuggle_sum=0x7f06013d;
+ public static final int init_autojuggle_sum=0x7f060140;
/** default number of minutes on timer for new games
*/
- public static final int initial_player_minutes=0x7f06013f;
+ public static final int initial_player_minutes=0x7f060142;
/**
*/
- public static final int invite_bt_desc_fmt=0x7f060217;
- public static final int invite_chooser_email=0x7f060190;
+ public static final int invite_bt_desc_fmt=0x7f06021a;
+ public static final int invite_chooser_email=0x7f060193;
/** When I've created the invitation, in text or html, I ask
Android to launch an app that can send it, typically an email
or messaging app. Android then asks the user to choose which
@@ -1204,20 +1205,20 @@ public final class R {
to have launched. This string is passed to Android and used
as the title of the dialog that presents that choice.
*/
- public static final int invite_chooser_fmt=0x7f06018f;
- public static final int invite_chooser_sms=0x7f060191;
- public static final int invite_dict_missing_body_fmt=0x7f060264;
- public static final int invite_dict_missing_body_noname_fmt=0x7f060265;
- public static final int invite_dict_missing_title=0x7f060263;
- public static final int invite_host=0x7f06006b;
+ public static final int invite_chooser_fmt=0x7f060192;
+ public static final int invite_chooser_sms=0x7f060194;
+ public static final int invite_dict_missing_body_fmt=0x7f060267;
+ public static final int invite_dict_missing_body_noname_fmt=0x7f060268;
+ public static final int invite_dict_missing_title=0x7f060266;
+ public static final int invite_host=0x7f06006c;
/** This is the body of the html version of the invitation. A URL
is created with parameters describing the game and
substituted for "%1$s". (The funky \u003c and friends are
encodings for the greater-than and less-than symbols which
are not legal in xml strings.)
*/
- public static final int invite_htm_fmt=0x7f06018d;
- public static final int invite_mime=0x7f06006d;
+ public static final int invite_htm_fmt=0x7f060190;
+ public static final int invite_mime=0x7f06006e;
/** If a networked game is opened and is not complete, i.e. if it
is listed as expecting remote players who have not yet shown
up, then the most likely explanation is that none has been
@@ -1227,48 +1228,48 @@ public final class R {
invititation while trying to warn him not to send duplicates.
The number of players missing is substituted for "%1$d".
*/
- public static final int invite_msg_fmt=0x7f0600f0;
+ public static final int invite_msg_fmt=0x7f0600f3;
/** Most users create games with only two players, which is the
default, but Crosswords supports up to four. When I'm using
the above string to encourage the opener of a game missing
players to invite, IF the number of missing players is
greater than one this text is appended to the above.
*/
- public static final int invite_multiple=0x7f0600f2;
- public static final int invite_prefix=0x7f06006c;
+ public static final int invite_multiple=0x7f0600f5;
+ public static final int invite_prefix=0x7f06006d;
/**
*/
- public static final int invite_progress=0x7f06020d;
+ public static final int invite_progress=0x7f060210;
/**
*/
- public static final int invite_sms_desc_fmt=0x7f060229;
+ public static final int invite_sms_desc_fmt=0x7f06022c;
/** This is the subject line of the email/text sent to invite
someone to join a game.
*/
- public static final int invite_subject_fmt=0x7f06018c;
+ public static final int invite_subject_fmt=0x7f06018f;
/**
*/
- public static final int invite_success=0x7f06022f;
+ public static final int invite_success=0x7f060232;
/** This is the body of the text version of the invitation. A URL
is created with parameters describing the game and
substituted for "%1$s".
*/
- public static final int invite_txt_fmt=0x7f06018e;
+ public static final int invite_txt_fmt=0x7f060191;
/** text of checkbox asking if user wants to search for open
public rooms
*/
- public static final int join_room=0x7f0600c6;
+ public static final int join_room=0x7f0600c9;
/** Checkbox that when set keeps the device screen from dimming
if the board screen is what's displayed. This is to allow
users to think about a move without having to touch the
screen all the time.
*/
- public static final int keep_screenon=0x7f06014c;
+ public static final int keep_screenon=0x7f06014f;
/** clarification of above
*/
- public static final int keep_screenon_summary=0x7f06014d;
+ public static final int keep_screenon_summary=0x7f060150;
public static final int key_background=0x7f060017;
- public static final int key_board_size=0x7f060026;
+ public static final int key_board_size=0x7f060027;
public static final int key_bonus_l2x=0x7f060011;
public static final int key_bonus_l3x=0x7f060012;
public static final int key_bonus_w2x=0x7f060013;
@@ -1276,233 +1277,234 @@ public final class R {
/** the color of text, e.g. "2L", shown on a bonus square on the
board
*/
- public static final int key_bonushint=0x7f06015c;
- public static final int key_bt_addrs=0x7f06003b;
- public static final int key_bt_names=0x7f06003a;
- public static final int key_checked_sms=0x7f060042;
+ public static final int key_bonushint=0x7f06015f;
+ public static final int key_bt_addrs=0x7f06003c;
+ public static final int key_bt_names=0x7f06003b;
+ public static final int key_checked_sms=0x7f060043;
/** database keys whose entries aren't visible prefs
*/
- public static final int key_closed_langs=0x7f060039;
+ public static final int key_closed_langs=0x7f06003a;
public static final int key_clr_bonushint=0x7f060019;
public static final int key_clr_crosshairs=0x7f060018;
/** prefs keys
*/
public static final int key_color_tiles=0x7f060002;
- public static final int key_connect_frequency=0x7f060030;
- public static final int key_connstat_data=0x7f06003d;
- public static final int key_default_dict=0x7f060028;
- public static final int key_default_group=0x7f060043;
- public static final int key_default_loc=0x7f060038;
- public static final int key_default_phonies=0x7f06002e;
- public static final int key_default_robodict=0x7f06002d;
- public static final int key_default_timerenabled=0x7f06002f;
- public static final int key_dev_id=0x7f06003e;
+ public static final int key_connect_frequency=0x7f060031;
+ public static final int key_connstat_data=0x7f06003e;
+ public static final int key_default_dict=0x7f060029;
+ public static final int key_default_group=0x7f060044;
+ public static final int key_default_loc=0x7f060039;
+ public static final int key_default_phonies=0x7f06002f;
+ public static final int key_default_robodict=0x7f06002e;
+ public static final int key_default_timerenabled=0x7f060030;
+ public static final int key_dev_id=0x7f06003f;
public static final int key_dict_host=0x7f060020;
- public static final int key_download_path=0x7f060064;
+ public static final int key_download_path=0x7f060065;
public static final int key_empty=0x7f060016;
- public static final int key_enable_debug=0x7f060062;
- public static final int key_enable_dup_invite=0x7f060063;
- public static final int key_enable_sms=0x7f060033;
+ public static final int key_enable_debug=0x7f060063;
+ public static final int key_enable_dup_invite=0x7f060064;
+ public static final int key_enable_sms=0x7f060034;
public static final int key_explain_robot=0x7f060005;
- public static final int key_gcmvers_regid=0x7f06003f;
- public static final int key_group_posns=0x7f060044;
+ public static final int key_gcmvers_regid=0x7f060040;
+ public static final int key_group_posns=0x7f060045;
public static final int key_hide_crosshairs=0x7f060009;
public static final int key_hide_title=0x7f06000c;
public static final int key_hide_values=0x7f06000a;
- public static final int key_init_autojuggle=0x7f060025;
- public static final int key_init_hintsallowed=0x7f060023;
- public static final int key_init_nethintsallowed=0x7f060024;
- public static final int key_initial_player_minutes=0x7f060027;
- public static final int key_keep_screenon=0x7f060034;
- public static final int key_last_packet=0x7f060045;
+ public static final int key_init_autojuggle=0x7f060026;
+ public static final int key_init_hintsallowed=0x7f060024;
+ public static final int key_init_nethintsallowed=0x7f060025;
+ public static final int key_initial_player_minutes=0x7f060028;
+ public static final int key_keep_screenon=0x7f060035;
+ public static final int key_last_packet=0x7f060046;
public static final int key_logging_on=0x7f060021;
- public static final int key_na_browse=0x7f06005e;
- public static final int key_na_browseall=0x7f06005f;
- public static final int key_na_firefox=0x7f06005d;
- public static final int key_na_lookup=0x7f06005c;
- public static final int key_na_studycopy=0x7f060061;
- public static final int key_na_values=0x7f060060;
- public static final int key_notagain_arrow=0x7f060058;
- public static final int key_notagain_backclears=0x7f06004a;
- public static final int key_notagain_chat=0x7f06004b;
- public static final int key_notagain_conndall=0x7f060055;
- public static final int key_notagain_conndfirst=0x7f060056;
- public static final int key_notagain_conndmid=0x7f060057;
- public static final int key_notagain_done=0x7f060053;
- public static final int key_notagain_flip=0x7f060050;
- public static final int key_notagain_hintnext=0x7f06004e;
- public static final int key_notagain_hintprev=0x7f06004d;
- public static final int key_notagain_juggle=0x7f06004f;
- public static final int key_notagain_newfrom=0x7f06005a;
- public static final int key_notagain_newselect=0x7f060049;
- public static final int key_notagain_relay=0x7f06004c;
- public static final int key_notagain_sms_ready=0x7f060048;
- public static final int key_notagain_sync=0x7f060047;
- public static final int key_notagain_trading=0x7f06005b;
- public static final int key_notagain_turnchanged=0x7f060059;
- public static final int key_notagain_undo=0x7f060052;
- public static final int key_notagain_unlock=0x7f060054;
- public static final int key_notagain_zoom=0x7f060051;
- public static final int key_notify_sound=0x7f060031;
- public static final int key_notify_vibrate=0x7f060032;
+ public static final int key_na_browse=0x7f06005f;
+ public static final int key_na_browseall=0x7f060060;
+ public static final int key_na_firefox=0x7f06005e;
+ public static final int key_na_lookup=0x7f06005d;
+ public static final int key_na_studycopy=0x7f060062;
+ public static final int key_na_values=0x7f060061;
+ public static final int key_notagain_arrow=0x7f060059;
+ public static final int key_notagain_backclears=0x7f06004b;
+ public static final int key_notagain_chat=0x7f06004c;
+ public static final int key_notagain_conndall=0x7f060056;
+ public static final int key_notagain_conndfirst=0x7f060057;
+ public static final int key_notagain_conndmid=0x7f060058;
+ public static final int key_notagain_done=0x7f060054;
+ public static final int key_notagain_flip=0x7f060051;
+ public static final int key_notagain_hintnext=0x7f06004f;
+ public static final int key_notagain_hintprev=0x7f06004e;
+ public static final int key_notagain_juggle=0x7f060050;
+ public static final int key_notagain_newfrom=0x7f06005b;
+ public static final int key_notagain_newselect=0x7f06004a;
+ public static final int key_notagain_relay=0x7f06004d;
+ public static final int key_notagain_sms_ready=0x7f060049;
+ public static final int key_notagain_sync=0x7f060048;
+ public static final int key_notagain_trading=0x7f06005c;
+ public static final int key_notagain_turnchanged=0x7f06005a;
+ public static final int key_notagain_undo=0x7f060053;
+ public static final int key_notagain_unlock=0x7f060055;
+ public static final int key_notagain_zoom=0x7f060052;
+ public static final int key_notify_sound=0x7f060032;
+ public static final int key_notify_vibrate=0x7f060033;
public static final int key_peek_other=0x7f060008;
public static final int key_player0=0x7f06000d;
public static final int key_player1=0x7f06000e;
- public static final int key_player1_name=0x7f060029;
+ public static final int key_player1_name=0x7f06002a;
public static final int key_player2=0x7f06000f;
- public static final int key_player2_name=0x7f06002a;
+ public static final int key_player2_name=0x7f06002b;
public static final int key_player3=0x7f060010;
- public static final int key_player3_name=0x7f06002b;
- public static final int key_player4_name=0x7f06002c;
+ public static final int key_player3_name=0x7f06002c;
+ public static final int key_player4_name=0x7f06002d;
public static final int key_proxy_port=0x7f06001e;
public static final int key_relay_host=0x7f06001a;
public static final int key_relay_port=0x7f06001b;
- public static final int key_relay_regid=0x7f060040;
- public static final int key_relay_regid_ackd=0x7f060041;
+ public static final int key_relay_regid=0x7f060041;
+ public static final int key_relay_regid_ackd=0x7f060042;
public static final int key_ringer_zoom=0x7f06000b;
+ public static final int key_send_data_sms=0x7f060023;
public static final int key_show_arrow=0x7f060003;
public static final int key_show_sms=0x7f060022;
public static final int key_skip_confirm=0x7f060006;
- public static final int key_sms_phones=0x7f06003c;
+ public static final int key_sms_phones=0x7f06003d;
public static final int key_sms_port=0x7f06001f;
public static final int key_sort_tiles=0x7f060007;
public static final int key_square_tiles=0x7f060004;
- public static final int key_studyon=0x7f060036;
- public static final int key_summary_field=0x7f060037;
- public static final int key_thumbsize=0x7f060035;
+ public static final int key_studyon=0x7f060037;
+ public static final int key_summary_field=0x7f060038;
+ public static final int key_thumbsize=0x7f060036;
public static final int key_tile_back=0x7f060015;
- public static final int key_udp_interval=0x7f060046;
+ public static final int key_udp_interval=0x7f060047;
public static final int key_update_prerel=0x7f06001d;
public static final int key_update_url=0x7f06001c;
- public static final int key_xlations_enabled=0x7f060066;
- public static final int key_xlations_locale=0x7f060065;
+ public static final int key_xlations_enabled=0x7f060067;
+ public static final int key_xlations_locale=0x7f060066;
/** text of separator marking out the language area of the
dialog. First is used for single-device games, and second
for networked games (where players aren't allowed to have
different wordlists.)
*/
- public static final int lang_label=0x7f0600c3;
- public static final int langdict_label=0x7f0600c4;
- public static final int list_group_default=0x7f06026f;
- public static final int list_group_delete=0x7f06026d;
- public static final int list_group_movedown=0x7f060271;
- public static final int list_group_moveup=0x7f060270;
- public static final int list_group_rename=0x7f06026e;
+ public static final int lang_label=0x7f0600c6;
+ public static final int langdict_label=0x7f0600c7;
+ public static final int list_group_default=0x7f060272;
+ public static final int list_group_delete=0x7f060270;
+ public static final int list_group_movedown=0x7f060274;
+ public static final int list_group_moveup=0x7f060273;
+ public static final int list_group_rename=0x7f060271;
/** ############## menu items ##############
pulls up dialog to configure the selected game
*/
- public static final int list_item_config=0x7f06009d;
+ public static final int list_item_config=0x7f0600a0;
/** makes a copy of the selected game. This is currently
disabled for networked games since there would be problems if
two identically configured games started trying to talk to a
remote game that expected there was only one of them.
*/
- public static final int list_item_copy=0x7f0600a3;
+ public static final int list_item_copy=0x7f0600a6;
/** pulls up dialog to delete the selected game
*/
- public static final int list_item_delete=0x7f0600a0;
+ public static final int list_item_delete=0x7f0600a3;
/** pulls up dialog to change the group of the selected game
*/
- public static final int list_item_move=0x7f06009f;
+ public static final int list_item_move=0x7f0600a2;
/** creates a new game with all configuation copied from the
selected game
*/
- public static final int list_item_new_from=0x7f0600a2;
+ public static final int list_item_new_from=0x7f0600a5;
/** pulls up dialog to rename (change name of) the selected game
*/
- public static final int list_item_rename=0x7f06009e;
+ public static final int list_item_rename=0x7f0600a1;
/** pulls up dialog to reset the selected game, that is to remove
all moves so that it's the same as a newly created game
except for any configuration.
*/
- public static final int list_item_reset=0x7f0600a1;
+ public static final int list_item_reset=0x7f0600a4;
/** One of the strings used in the right column of the list of
installed wordlists to describe those that are part of
Crosswords and that cannot be uninstalled or moved.
loc_internal and loc_external are the other possible strings
in this column.
*/
- public static final int loc_builtin=0x7f0600ab;
+ public static final int loc_builtin=0x7f0600ae;
/** Used to describe wordlists that are in the Downloads
directory. Currently I don't look there so this is unused,
but I will eventually do so. This should be the same name as
the built-in Android web browser uses for the directory where
it saves files it downloads.
*/
- public static final int loc_downloads=0x7f0600ac;
+ public static final int loc_downloads=0x7f0600af;
/**
*/
- public static final int loc_external=0x7f0600b9;
+ public static final int loc_external=0x7f0600bc;
/** see move_dictf above
*/
- public static final int loc_internal=0x7f0600b8;
- public static final int loc_main_english=0x7f060068;
- public static final int loc_main_yourlang=0x7f0602a6;
- public static final int loc_menu_xlate=0x7f0602a7;
+ public static final int loc_internal=0x7f0600bb;
+ public static final int loc_main_english=0x7f060069;
+ public static final int loc_main_yourlang=0x7f0602a9;
+ public static final int loc_menu_xlate=0x7f0602aa;
/** This is the "hint" printed in light text in the empty player
name field
*/
- public static final int local_name_hint=0x7f06019d;
- public static final int logging_on=0x7f060079;
- public static final int logging_on_summary=0x7f06007a;
- public static final int lookup_title=0x7f0602a2;
+ public static final int local_name_hint=0x7f0601a0;
+ public static final int logging_on=0x7f06007a;
+ public static final int logging_on_summary=0x7f06007b;
+ public static final int lookup_title=0x7f0602a5;
/**
*/
- public static final int manual_owner_name=0x7f06022a;
+ public static final int manual_owner_name=0x7f06022d;
/**
*/
- public static final int max_len=0x7f060202;
+ public static final int max_len=0x7f060205;
/**
*/
- public static final int menu_chat=0x7f06023b;
+ public static final int menu_chat=0x7f06023e;
/**
*/
- public static final int menu_flip=0x7f060239;
+ public static final int menu_flip=0x7f06023c;
/**
*/
- public static final int menu_hint_next=0x7f060236;
+ public static final int menu_hint_next=0x7f060239;
/**
*/
- public static final int menu_hint_prev=0x7f060235;
+ public static final int menu_hint_prev=0x7f060238;
/**
*/
- public static final int menu_juggle=0x7f060238;
+ public static final int menu_juggle=0x7f06023b;
/** text of menu that brings up the Settings (preferences) dialog
*/
- public static final int menu_prefs=0x7f060097;
- public static final int menu_rateme=0x7f060292;
+ public static final int menu_prefs=0x7f06009a;
+ public static final int menu_rateme=0x7f060295;
/**
*/
- public static final int menu_toggle_values=0x7f06023c;
+ public static final int menu_toggle_values=0x7f06023f;
/**
*/
- public static final int menu_zoom=0x7f06023a;
+ public static final int menu_zoom=0x7f06023d;
/**
*/
- public static final int min_len=0x7f060201;
+ public static final int min_len=0x7f060204;
/** label for the field used to set the timer's inital value
*/
- public static final int minutes_label=0x7f0600d0;
- public static final int missing_dict_detail=0x7f060262;
- public static final int missing_dict_title=0x7f060261;
+ public static final int minutes_label=0x7f0600d3;
+ public static final int missing_dict_detail=0x7f060265;
+ public static final int missing_dict_title=0x7f060264;
/** Used as a substitute for the names of remote players when
they aren't available yet because the connection is not
complete. Displayed in the lists of players found in each
game listing.
*/
- public static final int missing_player=0x7f06008d;
+ public static final int missing_player=0x7f060090;
/** When the dicts_item_move menu is chosen, this text is used in
the confirmation dialog. The name of the selected wordlist
is substituted for %1$s. The strings loc_internal and
loc_internal are substitued for %2$s and %3$s (or vice-versa,
depending on the current location of the wordlist.)
*/
- public static final int move_dict_fmt=0x7f0600b7;
+ public static final int move_dict_fmt=0x7f0600ba;
/** Otherwise they're listed with this to give some indication of
how far along they are. I may list "tiles left" someday
instead...
*/
- public static final int moves_fmt=0x7f060093;
+ public static final int moves_fmt=0x7f060096;
/**
############################################################
# :Dialogs:
@@ -1515,7 +1517,7 @@ public final class R {
Text of dialog. Player name is substituted
*/
- public static final int msg_ask_password_fmt=0x7f0601b5;
+ public static final int msg_ask_password_fmt=0x7f0601b8;
/** When a game has been connected and the relay is notified that
a device in the game has deleted its part of the game, this
message is posted when you connect your end of it to the
@@ -1523,21 +1525,21 @@ public final class R {
game so you might as well delete it (unless you're saving it
for its history etc.)
*/
- public static final int msg_dev_deleted=0x7f0600fb;
+ public static final int msg_dev_deleted=0x7f0600fe;
/** (I believe this can no longer occur)
*/
- public static final int msg_dup_room=0x7f0600f9;
+ public static final int msg_dup_room=0x7f0600fc;
/** (I believe this can no longer occur)
*/
- public static final int msg_lost_other=0x7f0600fa;
+ public static final int msg_lost_other=0x7f0600fd;
/** (I believe this can no longer occur)
*/
- public static final int msg_no_room=0x7f0600f8;
+ public static final int msg_no_room=0x7f0600fb;
/** Text of "toast" shown when a game is notified by the relay
that all expected players have registered. At this point
play can begin.
*/
- public static final int msg_relay_all_here_fmt=0x7f0600f5;
+ public static final int msg_relay_all_here_fmt=0x7f0600f8;
/** Text of "toast" (mini window briefly displayed) shown when a
game first connects to the relay. The three substitutions
are the device's order within the game (e.g. 2), the name of
@@ -1545,66 +1547,66 @@ public final class R {
registered with the relay in this game. This should be seen
only once per game.
*/
- public static final int msg_relay_waiting_fmt=0x7f0600f4;
+ public static final int msg_relay_waiting_fmt=0x7f0600f7;
/** (I believe this can no longer occur)
*/
- public static final int msg_too_many=0x7f0600f7;
+ public static final int msg_too_many=0x7f0600fa;
/** Text of progress indicator shown while check is being conducted
*/
- public static final int msgs_progress=0x7f060099;
- public static final int name_dict_fmt=0x7f060082;
+ public static final int msgs_progress=0x7f06009c;
+ public static final int name_dict_fmt=0x7f060085;
/** text of checkbox. If this checkbox is checked, games created
for network play will by default have the hint feature
enabled.
*/
- public static final int nethints_allowed=0x7f0600ce;
+ public static final int nethints_allowed=0x7f0600d1;
/** clarification on hints_allowed, whether new NETWORKED games
will default to having the hint feature enabled(string
elsewhere in this file)
*/
- public static final int nethints_allowed_sum=0x7f06013b;
- public static final int netstats_title=0x7f06007e;
- public static final int network_behavior=0x7f06016e;
+ public static final int nethints_allowed_sum=0x7f06013e;
+ public static final int netstats_title=0x7f060081;
+ public static final int network_behavior=0x7f060171;
/** explanation of the above
*/
- public static final int network_behavior_summary=0x7f06016f;
+ public static final int network_behavior_summary=0x7f060172;
/**
*/
- public static final int new_app_avail=0x7f060259;
+ public static final int new_app_avail=0x7f06025c;
/**
*/
- public static final int new_app_avail_fmt=0x7f060258;
+ public static final int new_app_avail_fmt=0x7f06025b;
/**
*/
- public static final int new_bt_body_fmt=0x7f060213;
+ public static final int new_bt_body_fmt=0x7f060216;
/**
*/
- public static final int new_bt_title=0x7f060212;
+ public static final int new_bt_title=0x7f060215;
/**
*/
- public static final int new_btmove_title=0x7f060215;
+ public static final int new_btmove_title=0x7f060218;
/**
*/
- public static final int new_dict_avail=0x7f060256;
+ public static final int new_dict_avail=0x7f060259;
/**
*/
- public static final int new_dict_avail_fmt=0x7f060257;
+ public static final int new_dict_avail_fmt=0x7f06025a;
/**
*/
- public static final int new_move_body=0x7f060216;
+ public static final int new_move_body=0x7f060219;
/**
*/
- public static final int new_name_body_fmt=0x7f060226;
+ public static final int new_name_body_fmt=0x7f060229;
/** hint (text shown when field is empty) for room name field
*/
- public static final int new_room_hint=0x7f0600c7;
+ public static final int new_room_hint=0x7f0600ca;
/**
*/
- public static final int new_sms_title=0x7f060225;
+ public static final int new_sms_title=0x7f060228;
/**
*/
- public static final int new_smsmove_title=0x7f060227;
- public static final int new_xlations_fmt=0x7f0602a8;
+ public static final int new_smsmove_title=0x7f06022a;
+ public static final int new_xlations_fmt=0x7f0602ab;
/**
###########################################################
# :Dialogs:
@@ -1617,36 +1619,36 @@ public final class R {
Title of New user info dialog
*/
- public static final int newbie_title=0x7f0601ca;
+ public static final int newbie_title=0x7f0601cd;
/** section separator (white-on-gray bar) for third section:
bluetooth games
*/
- public static final int newgame_bt_header=0x7f060183;
- public static final int newgame_enable_bt=0x7f060209;
+ public static final int newgame_bt_header=0x7f060186;
+ public static final int newgame_enable_bt=0x7f06020c;
/**
*/
- public static final int newgame_enable_sms=0x7f060223;
+ public static final int newgame_enable_sms=0x7f060226;
/** Text of first of two buttons for new networked games. Tap
this and a game will be created, but you probably won't see
it immediately because an email or messaging app will be
launched to send your invitation.
*/
- public static final int newgame_invite=0x7f060181;
+ public static final int newgame_invite=0x7f060184;
/** Text of first of two buttons for new standalone games. Tap
this and a new game will be created and opened. If the first
player is a robot it will immediately take its turn.
*/
- public static final int newgame_local=0x7f06017d;
+ public static final int newgame_local=0x7f060180;
/** Text of second of two buttons for new standalone games. Tap
this and you'll get taken to the "Game configure" screen
*/
- public static final int newgame_local_config=0x7f06017e;
+ public static final int newgame_local_config=0x7f060181;
/** This is one of two descriptions on this screen. It explains
what standalone games are and describes the two buttons used
to create them. The name of the language of the default
wordlist is substituted in for %1$s.
*/
- public static final int newgame_local_desc_fmt=0x7f06017c;
+ public static final int newgame_local_desc_fmt=0x7f06017f;
/** ############################################################
# :Screens:
# New game screen
@@ -1660,37 +1662,37 @@ public final class R {
section separator (white-on-gray bar) for first section:
standalone games
*/
- public static final int newgame_local_header=0x7f06017b;
+ public static final int newgame_local_header=0x7f06017e;
/** Text of second of two buttons for new networked games. Tap
this and you'll get taken to the "Game configure" screen
*/
- public static final int newgame_net_config=0x7f060182;
+ public static final int newgame_net_config=0x7f060185;
/** This is the second of two descriptions on this screen. It
explains what networked games are and describes the two
buttons used to create them.
*/
- public static final int newgame_networked_desc=0x7f060180;
+ public static final int newgame_networked_desc=0x7f060183;
/** section separator (white-on-gray bar) for second section:
networked games
*/
- public static final int newgame_networked_header=0x7f06017f;
+ public static final int newgame_networked_header=0x7f060182;
/**
*/
- public static final int newgame_sms_header=0x7f060220;
- public static final int newgroup_label=0x7f06026c;
- public static final int nfc_or_email=0x7f060187;
- public static final int nfc_or_sms_or_email=0x7f060188;
+ public static final int newgame_sms_header=0x7f060223;
+ public static final int newgroup_label=0x7f06026f;
+ public static final int nfc_or_email=0x7f06018a;
+ public static final int nfc_or_sms_or_email=0x7f06018b;
/** Title of dialog for renaming game (triggered by selecting
list_item_rename)
If you try to copy a networked game you get this error
message.
*/
- public static final int no_copy_network=0x7f0600a4;
+ public static final int no_copy_network=0x7f0600a7;
/** If the wordlist disappears mid-game there are no choices,
just an explanation and this button, after which the game
closes.
*/
- public static final int no_dict_finish=0x7f0601ae;
+ public static final int no_dict_finish=0x7f0601b1;
/** If the missing wordlist is discovered when trying to open the
game, we have more options. If there's another wordlist in
the same language, we can offer to substitute without needing
@@ -1699,12 +1701,12 @@ public final class R {
takes wordlist name and language substituted in for %1$ and
%2$
*/
- public static final int no_dict_fmt=0x7f0601b0;
+ public static final int no_dict_fmt=0x7f0601b3;
/** This is an alternative message presented when there's also
the option of downloading another wordlist. Game name,
wordlist name and language are substituted in.
*/
- public static final int no_dict_subst_fmt=0x7f0601b1;
+ public static final int no_dict_subst_fmt=0x7f0601b4;
/**
############################################################
# :Dialogs:
@@ -1716,32 +1718,32 @@ public final class R {
title of alert
*/
- public static final int no_dict_title=0x7f0601ad;
+ public static final int no_dict_title=0x7f0601b0;
/** If you tap the download_dicts button and Android fails to
find and launch an app that's registered to handle http
downloads a Toast is posted with this message.
*/
- public static final int no_download_warning=0x7f0600ad;
+ public static final int no_download_warning=0x7f0600b0;
/** If you click on the Play button without having entered a room
name you get an alert with this error message.
*/
- public static final int no_empty_rooms=0x7f06019f;
+ public static final int no_empty_rooms=0x7f0601a2;
/** If you choose the above option and have no networked games
you get this error message
*/
- public static final int no_games_to_refresh=0x7f06009a;
- public static final int no_hide_titlebar=0x7f06028d;
- public static final int no_market=0x7f060293;
- public static final int no_move_onegroup=0x7f060279;
+ public static final int no_games_to_refresh=0x7f06009d;
+ public static final int no_hide_titlebar=0x7f060290;
+ public static final int no_market=0x7f060296;
+ public static final int no_move_onegroup=0x7f06027c;
/** displayed when you long-tap a scoreboard entry and there's no
most recent score to show
*/
- public static final int no_moves_made=0x7f0600ef;
+ public static final int no_moves_made=0x7f0600f2;
/** If the query returns no rooms, this message is displayed,
with the number of players in the game and its language
substituted.
*/
- public static final int no_name_found_fmt=0x7f0600cb;
+ public static final int no_name_found_fmt=0x7f0600ce;
/** This is not currently shown
Crosswords wordlists, which are
just compressed lists of words plus tile information, determine
@@ -1754,49 +1756,49 @@ public final class R {
figuring out how to play and when you tap an empty cell the
arrow appears. This explains it.
*/
- public static final int not_again_arrow=0x7f0601dd;
- public static final int not_again_backclears=0x7f060283;
+ public static final int not_again_arrow=0x7f0601e0;
+ public static final int not_again_backclears=0x7f060286;
/**
*/
- public static final int not_again_browse=0x7f0601fe;
+ public static final int not_again_browse=0x7f060201;
/**
*/
- public static final int not_again_browseall=0x7f0601ff;
+ public static final int not_again_browseall=0x7f060202;
/** Shown when you tap the chat button on the toolbar of the
main Board screen
*/
- public static final int not_again_chat=0x7f0601d6;
+ public static final int not_again_chat=0x7f0601d9;
/** This is shown in the Board screen when you successfully
connecting a game to the relay and are the last device in the
game to do so, i.e. the game is now complete and you should
expect play to begin.
*/
- public static final int not_again_conndall=0x7f0601dc;
+ public static final int not_again_conndall=0x7f0601df;
/** This is shown in the Board screen when you successfully
connect a game to the relay and are the first device in the
game to do so.
*/
- public static final int not_again_conndfirst=0x7f0601da;
+ public static final int not_again_conndfirst=0x7f0601dd;
/** This is shown in the Board screen when you successfully
connecting a game to the relay and are not the first device
in the game but not the last either. So it will only occur
for games with more than two devices, which are rare.
*/
- public static final int not_again_conndmid=0x7f0601db;
+ public static final int not_again_conndmid=0x7f0601de;
/** This is shown when you choose the board_menu_done menu item.
It's to let you know that there's a shortcut that does almost
the same thing.
*/
- public static final int not_again_done=0x7f0601d8;
- public static final int not_again_firefox=0x7f060291;
+ public static final int not_again_done=0x7f0601db;
+ public static final int not_again_firefox=0x7f060294;
/** Shown when you tap the flip button on the toolbar of the main
Board screen
*/
- public static final int not_again_flip=0x7f0601d3;
+ public static final int not_again_flip=0x7f0601d6;
/** Shown when you tap the next hint button on the toolbar of the
main Board screen
*/
- public static final int not_again_hintnext=0x7f0601d1;
+ public static final int not_again_hintnext=0x7f0601d4;
/** Currently not used
The new game you have created has
two players. Player 1 is a robot; Player 2 is you. Tap the game
@@ -1805,75 +1807,75 @@ public final class R {
Shown when you tap the Previous Hint button on the toolbar of
the main Board screen
*/
- public static final int not_again_hintprev=0x7f0601d0;
+ public static final int not_again_hintprev=0x7f0601d3;
/** Shown when you tap the juggle button on the toolbar of the
main Board screen
*/
- public static final int not_again_juggle=0x7f0601d2;
+ public static final int not_again_juggle=0x7f0601d5;
/**
*/
- public static final int not_again_lookup=0x7f0601f3;
+ public static final int not_again_lookup=0x7f0601f6;
/** Shown when you first pick the list_item_new_from menuitem
*/
- public static final int not_again_newfrom=0x7f0601df;
- public static final int not_again_newselect=0x7f060282;
- public static final int not_again_sms_ready=0x7f06028f;
- public static final int not_again_studycopy=0x7f0602a1;
+ public static final int not_again_newfrom=0x7f0601e2;
+ public static final int not_again_newselect=0x7f060285;
+ public static final int not_again_sms_ready=0x7f060292;
+ public static final int not_again_studycopy=0x7f0602a4;
/** The following strings (all whose names start with
"not_again") appear in the New user info dialog.
shown when user chooses the gamel_menu_checkmoves menu
*/
- public static final int not_again_sync=0x7f0601cc;
+ public static final int not_again_sync=0x7f0601cf;
/** Shown when the user chooses the "board_menu_trade" menu
*/
- public static final int not_again_trading=0x7f0601cd;
- public static final int not_again_trading_buttons=0x7f0601ce;
- public static final int not_again_trading_menu=0x7f0601cf;
+ public static final int not_again_trading=0x7f0601d0;
+ public static final int not_again_trading_buttons=0x7f0601d1;
+ public static final int not_again_trading_menu=0x7f0601d2;
/** Shown when the board screen is visible and it's just become
another players turn. The idea is to give a hint about how to
find out about recent moves.
*/
- public static final int not_again_turnchanged=0x7f0601de;
+ public static final int not_again_turnchanged=0x7f0601e1;
/** Shown when you tap the undo/redo button on the toolbar of the
main Board screen
*/
- public static final int not_again_undo=0x7f0601d5;
+ public static final int not_again_undo=0x7f0601d8;
/** Shown in the Game configure screen when the game_locked
checkbox is checked and you uncheck it.
*/
- public static final int not_again_unlock=0x7f0601d9;
+ public static final int not_again_unlock=0x7f0601dc;
/** Shown when you tap the values button on the toolbar of the
main Board screen. This is intended to allow players to
remind themselves how much played tiles are worth while
planning a move.
*/
- public static final int not_again_values=0x7f0601d7;
+ public static final int not_again_values=0x7f0601da;
/** Shown when you tap the zoom (+/-) button on the toolbar of
the main Board screen
*/
- public static final int not_again_zoom=0x7f0601d4;
- public static final int notify_body_fmt=0x7f0601b9;
+ public static final int not_again_zoom=0x7f0601d7;
+ public static final int notify_body_fmt=0x7f0601bc;
/** This text is displayed as the "summary" for both of the
notify choices above, as extra information. (There could be
different strings if it makes more sense in your language.)
*/
- public static final int notify_other_summary=0x7f06017a;
+ public static final int notify_other_summary=0x7f06017d;
/** When one or more new moves is found, should I play a
notification sound
*/
- public static final int notify_sound=0x7f060178;
+ public static final int notify_sound=0x7f06017b;
/** When a move is fetched from the relay a Notification is
posted. These are its title, which appears in the top bar of the
device, and the body that appears when you pull the notifications
down.
*/
- public static final int notify_title=0x7f0601b8;
+ public static final int notify_title=0x7f0601bb;
/** When one or more new moves is found, should I vibrate the
device
*/
- public static final int notify_vibrate=0x7f060179;
+ public static final int notify_vibrate=0x7f06017c;
/** text of label identifying the field where human players can
enter an option password. The label and field disappear when
the robot player checkbox is checked because it makes no
@@ -1882,9 +1884,9 @@ public final class R {
player on a device, so they are infrequently used in network
games as well.
*/
- public static final int password_label=0x7f0600e7;
- public static final int paste_done_fmt=0x7f06029b;
- public static final int pct_suffix=0x7f060290;
+ public static final int password_label=0x7f0600ea;
+ public static final int paste_done_fmt=0x7f06029e;
+ public static final int pct_suffix=0x7f060293;
/** If this preference is checked, tapping on the scoreboard
entry for any player reveals that player's tiles and any
pending move (after asking for his password if one is set.)
@@ -1894,57 +1896,57 @@ public final class R {
though it's not his turn e.g. while the player whose turn it
is is temporarily unable to play.
*/
- public static final int peek_other=0x7f06016a;
+ public static final int peek_other=0x7f06016d;
/** explanation of the above
*/
- public static final int peek_other_summary=0x7f06016b;
+ public static final int peek_other_summary=0x7f06016e;
/**
*/
- public static final int phone_label=0x7f060231;
+ public static final int phone_label=0x7f060234;
/** Don't warn, but simply force to skip turn (give 0 points)
when user attempts to play word not in the wordlist.
*/
- public static final int phonies_disallow=0x7f0600d8;
+ public static final int phonies_disallow=0x7f0600db;
/** These are the three choices in the popup above whose text is
phonies_spinner_prompt
Don't care if words played are in the wordlist or not
*/
- public static final int phonies_ignore=0x7f0600d6;
+ public static final int phonies_ignore=0x7f0600d9;
/** title of popup used to determine how words are handled that
are not in the wordlist used for the game (or player if using
different wordlists per player)
*/
- public static final int phonies_spinner_prompt=0x7f0600d5;
+ public static final int phonies_spinner_prompt=0x7f0600d8;
/** warn player when word played is not in the wordlist, but
allow him to play it.
*/
- public static final int phonies_warn=0x7f0600d7;
+ public static final int phonies_warn=0x7f0600da;
/**
*/
- public static final int pick_faceup=0x7f0601fa;
+ public static final int pick_faceup=0x7f0601fd;
/**
*/
- public static final int pick_url_title_fmt=0x7f0601f1;
+ public static final int pick_url_title_fmt=0x7f0601f4;
/** Text of button at bottom. Press it and you exit this
configure screen and open the game into the Board screen
*/
- public static final int play=0x7f0600dd;
+ public static final int play=0x7f0600e0;
/** The remaining strings (down to the color edit dialog below)
are showns as the names of editable colors and as the the
title of the color editor that comes up when the name is
tapped.
First of four players (color for, that is )
*/
- public static final int player0=0x7f060150;
+ public static final int player0=0x7f060153;
/** Second of four players (color for, that is )
*/
- public static final int player1=0x7f060151;
+ public static final int player1=0x7f060154;
/** Third of four players (color for, that is )
*/
- public static final int player2=0x7f060152;
+ public static final int player2=0x7f060155;
/** Fourth of four players (color for, that is )
*/
- public static final int player3=0x7f060153;
+ public static final int player3=0x7f060156;
/**
############################################################
# :Dialogs:
@@ -1956,40 +1958,40 @@ public final class R {
dialog title
*/
- public static final int player_edit_title=0x7f0600e1;
+ public static final int player_edit_title=0x7f0600e4;
/** used to create default player names. Number between 1 and 4
is substituted
*/
- public static final int player_fmt=0x7f0601b7;
+ public static final int player_fmt=0x7f0601ba;
/** text of lable identifying the field in which player's name is
set/displayed
*/
- public static final int player_label=0x7f0600e3;
+ public static final int player_label=0x7f0600e6;
/** text for separator above the list of players that's used for
networked games. The numbers of local and non-local players
are substituted for %1$d and %2$d.
*/
- public static final int players_label_host_fmt=0x7f0600c0;
+ public static final int players_label_host_fmt=0x7f0600c3;
/** text for separator above the list of players that's used for
non-networked games
*/
- public static final int players_label_standalone=0x7f0600bf;
+ public static final int players_label_standalone=0x7f0600c2;
/** Toast shown when players are missing, same time as above
but more subtle.
*/
- public static final int players_miss_fmt=0x7f0600f1;
+ public static final int players_miss_fmt=0x7f0600f4;
/** Label for the first "player name" preference
*/
- public static final int pref_player1_name=0x7f060131;
+ public static final int pref_player1_name=0x7f060134;
/** Label for the second "player name" preference (unused)
*/
- public static final int pref_player2_name=0x7f060132;
+ public static final int pref_player2_name=0x7f060135;
/** Label for the third "player name" preference (unused)
*/
- public static final int pref_player3_name=0x7f060133;
+ public static final int pref_player3_name=0x7f060136;
/** Label for the fourth "player name" preference (unused)
*/
- public static final int pref_player4_name=0x7f060134;
+ public static final int pref_player4_name=0x7f060137;
/**
############################################################
# :Screens:
@@ -1998,10 +2000,10 @@ public final class R {
title of this sub-preference
*/
- public static final int prefs_appearance=0x7f060141;
+ public static final int prefs_appearance=0x7f060144;
/** clarification of the above
*/
- public static final int prefs_appearance_summary=0x7f060142;
+ public static final int prefs_appearance_summary=0x7f060145;
/**
############################################################
# :Screens:
@@ -2010,10 +2012,10 @@ public final class R {
title of this sub-preference
*/
- public static final int prefs_behavior=0x7f060160;
+ public static final int prefs_behavior=0x7f060163;
/** clarification of the above
*/
- public static final int prefs_behavior_summary=0x7f060161;
+ public static final int prefs_behavior_summary=0x7f060164;
/**
############################################################
# :Screens:
@@ -2023,10 +2025,10 @@ public final class R {
title of this sub-preference
*/
- public static final int prefs_colors=0x7f06014e;
+ public static final int prefs_colors=0x7f060151;
/** clarification of the above
*/
- public static final int prefs_colors_summary=0x7f06014f;
+ public static final int prefs_colors_summary=0x7f060152;
/**
############################################################
# :Screens:
@@ -2035,49 +2037,49 @@ public final class R {
title of this sub-preference
*/
- public static final int prefs_defaults=0x7f06012d;
+ public static final int prefs_defaults=0x7f060130;
/** clarification of the above
*/
- public static final int prefs_defaults_summary=0x7f06012e;
+ public static final int prefs_defaults_summary=0x7f060131;
/** sub-preference for dictionaries (soon to be called "word lists")
*/
- public static final int prefs_dicts=0x7f060136;
+ public static final int prefs_dicts=0x7f060139;
/** clarification of above
*/
- public static final int prefs_dicts_summary=0x7f060137;
+ public static final int prefs_dicts_summary=0x7f06013a;
/** sub-preference title for editing default player names.
There's only enabled now, though.
*/
- public static final int prefs_names=0x7f06012f;
+ public static final int prefs_names=0x7f060132;
/** clarification of the above
*/
- public static final int prefs_names_summary=0x7f060130;
+ public static final int prefs_names_summary=0x7f060133;
/**
*/
- public static final int prompt_max_len=0x7f060204;
+ public static final int prompt_max_len=0x7f060207;
/**
*/
- public static final int prompt_min_len=0x7f060203;
- public static final int proxy_port=0x7f060081;
+ public static final int prompt_min_len=0x7f060206;
+ public static final int proxy_port=0x7f060084;
/** Short for "points", this is shown at the right end of the
tray in place of the first tile placed along with the points
the current move would earn if committed.
*/
- public static final int pts=0x7f0600f3;
+ public static final int pts=0x7f0600f6;
/** If you check the join_room checkbox Crosswords queries the
relay. This is the text of the progress indicator displayed what
that's going on. (It's often visible for only a very short
time.)
*/
- public static final int public_names_progress_fmt=0x7f0600ca;
+ public static final int public_names_progress_fmt=0x7f0600cd;
/** Title for generic dialog asking a question, usually in the
middle of a game, like "do you want to commit this move?"
*/
- public static final int query_title=0x7f0601c9;
+ public static final int query_title=0x7f0601cc;
/** Text of dialog asking user to confirm a move that exchanges
tiles (instead of forming a new word to earn points)
*/
- public static final int query_trade_fmt=0x7f060192;
+ public static final int query_trade_fmt=0x7f060195;
/**
############################################################
# :Dialogs:
@@ -2090,13 +2092,13 @@ public final class R {
Names of the three colors by which colors can be edited in
the color preferences dialog
*/
- public static final int red=0x7f06015d;
- public static final int redir_host=0x7f060077;
+ public static final int red=0x7f060160;
+ public static final int redir_host=0x7f060078;
/** Title of dialog used to alert players to relay-related
problems with the current game.
*/
- public static final int relay_alert=0x7f0600f6;
- public static final int relay_alert_title=0x7f060281;
+ public static final int relay_alert=0x7f0600f9;
+ public static final int relay_alert_title=0x7f060284;
/** ############################################################
# :Screens:
# RelayGameActivity screen
@@ -2117,63 +2119,63 @@ public final class R {
This is the heading above the list of steps. The default
language is substituted for %1$s.
*/
- public static final int relay_game_explain_fmt=0x7f060198;
- public static final int relay_host=0x7f060076;
- public static final int relay_port=0x7f060080;
+ public static final int relay_game_explain_fmt=0x7f06019b;
+ public static final int relay_host=0x7f060077;
+ public static final int relay_port=0x7f060083;
/** Shown in toast when relaunching after switching dicts
*/
- public static final int reload_new_dict_fmt=0x7f060260;
+ public static final int reload_new_dict_fmt=0x7f060263;
/** checkbox determining of this player is on this device or
remote. If remote, then the rest of the fields disappear
(since they will be set by the remote device.)
*/
- public static final int remote_label=0x7f0600e2;
+ public static final int remote_label=0x7f0600e5;
/**
*/
- public static final int remote_undone=0x7f060234;
- public static final int rename_group_label=0x7f060276;
+ public static final int remote_undone=0x7f060237;
+ public static final int rename_group_label=0x7f060279;
/** text within rename dialog (triggered by selecting
list_item_rename)
*/
- public static final int rename_label=0x7f0600a6;
- public static final int resend_finished_fmt=0x7f06010c;
+ public static final int rename_label=0x7f0600a9;
+ public static final int resend_finished_fmt=0x7f06010f;
/** if this preference is checked, the hardware volume keys will
work to zoom the board in and out (and will not control
volume) This only applies when the Board screen is
frontmost.
*/
- public static final int ringer_zoom=0x7f060168;
+ public static final int ringer_zoom=0x7f06016b;
/** explanation of the above
*/
- public static final int ringer_zoom_summary=0x7f060169;
+ public static final int ringer_zoom_summary=0x7f06016c;
/** checkbox determining if player is robot/automated or human
*/
- public static final int robot_label=0x7f0600e6;
+ public static final int robot_label=0x7f0600e9;
/** Used to format robot player names in the lists of players
found in each game listing
*/
- public static final int robot_name_fmt=0x7f06008c;
+ public static final int robot_name_fmt=0x7f06008f;
/** Three possible choices presented in the popup above
*/
- public static final int robot_smart=0x7f0600d2;
- public static final int robot_smarter=0x7f0600d3;
- public static final int robot_smartest=0x7f0600d4;
+ public static final int robot_smart=0x7f0600d5;
+ public static final int robot_smarter=0x7f0600d6;
+ public static final int robot_smartest=0x7f0600d7;
/** title of popup used to select how "smart" (how capable) the
robot player will be.
*/
- public static final int robot_spinner_prompt=0x7f0600d1;
+ public static final int robot_spinner_prompt=0x7f0600d4;
/** text of checkbox
*/
- public static final int room_public=0x7f0600c8;
+ public static final int room_public=0x7f0600cb;
/** title for popup of public rooms found on server
*/
- public static final int room_public_prompt=0x7f0600c9;
+ public static final int room_public_prompt=0x7f0600cc;
/**
*/
- public static final int scan_progress=0x7f06020c;
- public static final int sel_games_fmt=0x7f060284;
- public static final int sel_groups_fmt=0x7f060285;
- public static final int sel_items_fmt=0x7f0602a5;
+ public static final int scan_progress=0x7f06020f;
+ public static final int sel_games_fmt=0x7f060287;
+ public static final int sel_groups_fmt=0x7f060288;
+ public static final int sel_items_fmt=0x7f0602a8;
/**
############################################################
# :Dialogs:
@@ -2184,67 +2186,67 @@ public final class R {
up when the dicts_item_select menuitem is chosen. The
possible answers are the three button text strings below.
*/
- public static final int set_default_message_fmt=0x7f0600b3;
+ public static final int set_default_message_fmt=0x7f0600b6;
/** text of separator marking out other-setting area of the dialog
*/
- public static final int settings_label=0x7f0600cc;
+ public static final int settings_label=0x7f0600cf;
/** Checkbox that when set makes taps on the board manipulate an
arrow that then directs where tiles go when tapped in the
tray.
*/
- public static final int show_arrow=0x7f06014a;
+ public static final int show_arrow=0x7f06014d;
/** clarification of above
*/
- public static final int show_arrow_summary=0x7f06014b;
- public static final int show_wordlist_browser=0x7f060280;
+ public static final int show_arrow_summary=0x7f06014e;
+ public static final int show_wordlist_browser=0x7f060283;
/** If this preference is checked, the user will not be asked to
confirm after selecting the "Turn done" menu (or tapping the
points display at the right end of the tray)
*/
- public static final int skip_confirm_turn=0x7f060164;
+ public static final int skip_confirm_turn=0x7f060167;
/** explanation of the above
*/
- public static final int skip_confirm_turn_summary=0x7f060165;
- public static final int slmenu_clear_sel=0x7f060299;
- public static final int slmenu_copy_sel=0x7f060298;
- public static final int slmenu_deselect_all=0x7f0602a4;
- public static final int slmenu_select_all=0x7f0602a3;
+ public static final int skip_confirm_turn_summary=0x7f060168;
+ public static final int slmenu_clear_sel=0x7f06029c;
+ public static final int slmenu_copy_sel=0x7f06029b;
+ public static final int slmenu_deselect_all=0x7f0602a7;
+ public static final int slmenu_select_all=0x7f0602a6;
/**
*/
- public static final int sms_disabled=0x7f060222;
+ public static final int sms_disabled=0x7f060225;
/**
*/
- public static final int sms_networked_desc=0x7f060221;
+ public static final int sms_networked_desc=0x7f060224;
/** The invitation process begins with this query. The choice is
between html and plaintext formatting but I also provide some
explanation/guidance.
*/
- public static final int sms_or_email=0x7f060186;
- public static final int sms_ready_text=0x7f06028e;
+ public static final int sms_or_email=0x7f060189;
+ public static final int sms_ready_text=0x7f060291;
/**
*/
- public static final int smsname_label=0x7f06021c;
- public static final int square_tiles=0x7f06027d;
- public static final int square_tiles_summary=0x7f06027e;
+ public static final int smsname_label=0x7f06021f;
+ public static final int square_tiles=0x7f060280;
+ public static final int square_tiles_summary=0x7f060281;
/** Used in formatting moves and history
*/
- public static final int str_bonus_all=0x7f060123;
+ public static final int str_bonus_all=0x7f060126;
/** Using the hint feature is cheating by some players, and it
can be disabled via the hints_allowed preference. I should
be disabling the buttons in this case, but if I don't and you
try to get a hint you'll get this message instead.
*/
- public static final int str_cant_hint_while_disabled=0x7f0601a9;
+ public static final int str_cant_hint_while_disabled=0x7f0601ac;
/** Displayed if you try to use the undo menuitem or button and
there are no tiles on the board (no move has yet been made.)
[If I'm being clever and disabling those features in this
case there may be no way to see this.]
*/
- public static final int str_cant_undo_tileassign=0x7f0601a8;
+ public static final int str_cant_undo_tileassign=0x7f0601ab;
/** Beginning of the message presented to a user when asking him
to confirm committing the current turn
*/
- public static final int str_commit_confirm=0x7f060120;
+ public static final int str_commit_confirm=0x7f060123;
/** Used to format game name plus some other information as the
one-line summary for each game in the main screen. The name
of the game is substituted for %1$s. Something else
@@ -2252,21 +2254,21 @@ public final class R {
does not require translation unless the parentheses or
ordering is wrong for your language.
*/
- public static final int str_game_name_fmt=0x7f06008b;
+ public static final int str_game_name_fmt=0x7f06008e;
/** Used in formatting history and move reports; means user
skipped a turn because of an attempted illegal move/play of a
phony
*/
- public static final int str_lostturn=0x7f06011f;
+ public static final int str_lostturn=0x7f060122;
/** Displayed when you try to commit a turn that is illegal
because there is empty space between some of the tiles
placed, i.e. they do not form a single word.
*/
- public static final int str_no_empties_in_turn=0x7f0601a1;
+ public static final int str_no_empties_in_turn=0x7f0601a4;
/** Same as above, but used when you try to show tiles belonging
to a player on another device (a remote player.)
*/
- public static final int str_no_peek_remote_tiles=0x7f0601a6;
+ public static final int str_no_peek_remote_tiles=0x7f0601a9;
/** Displayed when you try to reveal a robot player's tiles,
either by tapping on its "hidden" rack (marked by "?"
characters) or by tapping on its scoreboard entry. (The
@@ -2274,32 +2276,32 @@ public final class R {
controlled by the peek_other preference and is disabled by
default.)
*/
- public static final int str_no_peek_robot_tiles=0x7f0601a5;
+ public static final int str_no_peek_robot_tiles=0x7f0601a8;
/** Used, with remote player's name substituted for %1$s, to
indicate that the player is remote.
*/
- public static final int str_nonlocal_name_fmt=0x7f060121;
+ public static final int str_nonlocal_name_fmt=0x7f060124;
/** Displyed when you try to commit a move and it's not your
turn.
*/
- public static final int str_not_your_turn=0x7f0601a4;
+ public static final int str_not_your_turn=0x7f0601a7;
/** Used in formatting game history and move summaries
*/
- public static final int str_pass_fmt=0x7f060116;
+ public static final int str_pass_fmt=0x7f060119;
/** Used in formatting exchange move summaries: passed means the
user skipped his turn, or made a move involving 0 tiles for 0
points
*/
- public static final int str_passed=0x7f06011c;
+ public static final int str_passed=0x7f06011f;
/** Used to alert user to loss of turn when a move is made and
phonies is set to lose turn when word used not in wordlist
*/
- public static final int str_phony_rejected=0x7f06011b;
+ public static final int str_phony_rejected=0x7f06011e;
/** Shown when using the the Game configure screen to configure a
networked game and you try to make all players local.
*/
- public static final int str_reg_server_sans_remote=0x7f0600d9;
+ public static final int str_reg_server_sans_remote=0x7f0600dc;
/** This error message is shown when a remote device tries to
join a game and is providing more players than that game
expects. I do not believe it is possible to see this message
@@ -2307,13 +2309,13 @@ public final class R {
many players are expected and only connects devices where the
numbers match.
*/
- public static final int str_reg_unexpected_user=0x7f0601aa;
+ public static final int str_reg_unexpected_user=0x7f0601ad;
/** Used in formatting remote player move summaries
*/
- public static final int str_remote_moved_fmt=0x7f060114;
+ public static final int str_remote_moved_fmt=0x7f060117;
/** Used in formatting final scores display
*/
- public static final int str_resigned=0x7f06025a;
+ public static final int str_resigned=0x7f06025d;
/**
############################################################
# Board info/error dialog messages
@@ -2324,18 +2326,18 @@ public final class R {
Notifies user of a normal robot move
*/
- public static final int str_robot_moved=0x7f06010d;
+ public static final int str_robot_moved=0x7f060110;
/** Shown when a remote device tries to join a game and wants to
play in a different language. As with the above, this should
be impossible when connecting via the relay.
*/
- public static final int str_server_dict_wins=0x7f0601ab;
+ public static final int str_server_dict_wins=0x7f0601ae;
/** Displayed when you try to commit a turn that is illegal
because tiles played do not touch other tiles already on the
board (or the middle/star square in the case of the initial
move.)
*/
- public static final int str_tiles_must_contact=0x7f0601a3;
+ public static final int str_tiles_must_contact=0x7f0601a6;
/**
error messages
@@ -2343,89 +2345,89 @@ public final class R {
because the tiles placed are not all in the same row or
column.
*/
- public static final int str_tiles_not_in_line=0x7f0601a0;
+ public static final int str_tiles_not_in_line=0x7f0601a3;
/** Displayed when you try to begin a trade but there are not
seven or more tiles in the pool. The rules don't allow
trading in this case.
*/
- public static final int str_too_few_tiles_left_to_trade=0x7f0601a7;
+ public static final int str_too_few_tiles_left_to_trade=0x7f0601aa;
/** Displayed when you try to commit the first move of the game
and are playing a single tile. The first move must include
two or more tiles.
*/
- public static final int str_two_tiles_first_move=0x7f0601a2;
+ public static final int str_two_tiles_first_move=0x7f0601a5;
/** Used in formatting final scores display
*/
- public static final int str_winner=0x7f06025b;
+ public static final int str_winner=0x7f06025e;
/** Used in formatting game history (not move summaries since
information about the current rack is hidden then)
*/
- public static final int strd_cumulative_score_fmt=0x7f060113;
+ public static final int strd_cumulative_score_fmt=0x7f060116;
/** Used in formatting game history and move summaries
*/
- public static final int strd_remaining_tiles_add_fmt=0x7f060110;
+ public static final int strd_remaining_tiles_add_fmt=0x7f060113;
/** Intro to the paragraph lisiting all of the tiles remaining
*/
- public static final int strd_remains_expl_fmt=0x7f060126;
+ public static final int strd_remains_expl_fmt=0x7f060129;
/** First line in the remaining tiles dialog (reached by tapping
the number at left end of the scoreboard)
*/
- public static final int strd_remains_header_fmt=0x7f060125;
+ public static final int strd_remains_header_fmt=0x7f060128;
/** Notifies user of a robot trade move
*/
- public static final int strd_robot_traded_fmt=0x7f06010e;
+ public static final int strd_robot_traded_fmt=0x7f060111;
/** I don't know how this is used. :-)
*/
- public static final int strd_time_penalty_sub=0x7f060115;
+ public static final int strd_time_penalty_sub=0x7f060118;
/** Used in formatting reports of trades (exchanges of tiles).
Number of tiles traded is substituted for %1$d
*/
- public static final int strd_traded_fmt=0x7f06011e;
+ public static final int strd_traded_fmt=0x7f060121;
/** Used in formatting moves and history. The total score for
one turn is substituted for %1$d.
*/
- public static final int strd_turn_score_fmt=0x7f060124;
+ public static final int strd_turn_score_fmt=0x7f060127;
/** Used in formatting game history and move summaries
*/
- public static final int strd_unused_tiles_sub_fmt=0x7f060111;
+ public static final int strd_unused_tiles_sub_fmt=0x7f060114;
/** Used in formatting game history and move summaries
*/
- public static final int strs_move_across_fmt=0x7f060117;
+ public static final int strs_move_across_fmt=0x7f06011a;
/** Used in formatting game history and move summaries
*/
- public static final int strs_move_down_fmt=0x7f060118;
+ public static final int strs_move_down_fmt=0x7f06011b;
/** Used in formatting game history and move summaries
*/
- public static final int strs_new_tiles_fmt=0x7f060112;
+ public static final int strs_new_tiles_fmt=0x7f060115;
/** Used in formatting game history and move summaries
*/
- public static final int strs_tray_at_start_fmt=0x7f060119;
+ public static final int strs_tray_at_start_fmt=0x7f06011c;
/** title for window you get when you select menu with text
board_menu_game_counts
*/
- public static final int strs_values_header_fmt=0x7f06010f;
+ public static final int strs_values_header_fmt=0x7f060112;
/** formats tiles and score together, and probably doesn't need
translation unless the colon ':' needs to be replaced
*/
- public static final int strsd_summaryscored_fmt=0x7f06011d;
+ public static final int strsd_summaryscored_fmt=0x7f060120;
/** Used in formatting exchange move summaries
*/
- public static final int strss_traded_for_fmt=0x7f06011a;
- public static final int study_langpick=0x7f06029e;
- public static final int study_no_lang_fmt=0x7f06029f;
- public static final int study_no_lists=0x7f0602a0;
- public static final int studylist_title_fmt=0x7f06029d;
+ public static final int strss_traded_for_fmt=0x7f06011d;
+ public static final int study_langpick=0x7f0602a1;
+ public static final int study_no_lang_fmt=0x7f0602a2;
+ public static final int study_no_lists=0x7f0602a3;
+ public static final int studylist_title_fmt=0x7f0602a0;
/** If substituting an existing same-language wordlist by
choosing button_substdict user gets to choose from a list of
wordlists. This is the title of that list.
*/
- public static final int subst_dict_title=0x7f0601b4;
+ public static final int subst_dict_title=0x7f0601b7;
/**
*/
- public static final int summary_conn=0x7f060211;
+ public static final int summary_conn=0x7f060214;
/**
*/
- public static final int summary_conn_sms_fmt=0x7f060232;
+ public static final int summary_conn_sms_fmt=0x7f060235;
/** Preference to control what's listed next to game name in the
first line of a game summary in the main games-list screen,
e.g., if the option is "Game Language", "English" in the
@@ -2434,10 +2436,10 @@ public final class R {
The following strings beginning with "game_summary_field_" are
possible values
*/
- public static final int summary_field=0x7f060143;
+ public static final int summary_field=0x7f060146;
/**
*/
- public static final int summary_gameover=0x7f060210;
+ public static final int summary_gameover=0x7f060213;
/** The display of each networked game includes one of three
states it can be in in the process of connecting to the
relay. These next three strings are used to create those
@@ -2445,52 +2447,53 @@ public final class R {
First state: is configured to use a room but has not yet
contacted the relay and been assigned that room.
*/
- public static final int summary_relay_conf_fmt=0x7f06008e;
+ public static final int summary_relay_conf_fmt=0x7f060091;
/** Third state: enough devices have connected in the room to
form a complete game. We'll be in this state as long as the
game exists.
*/
- public static final int summary_relay_conn_fmt=0x7f060090;
+ public static final int summary_relay_conn_fmt=0x7f060093;
/** Final state: game is over.
*/
- public static final int summary_relay_gameover_fmt=0x7f060091;
+ public static final int summary_relay_gameover_fmt=0x7f060094;
/** Second state: has been assigned to a room on the relay
(meaning the network is working) but there are not yet as
many players as expected: the game is not complete and play
cannot happen.
*/
- public static final int summary_relay_wait_fmt=0x7f06008f;
+ public static final int summary_relay_wait_fmt=0x7f060092;
+ public static final int summary_send_data_sms=0x7f06007f;
/** explanation of the above
*/
- public static final int summary_sort_tiles=0x7f060167;
- public static final int summary_studyon=0x7f060296;
- public static final int summary_thumbsize=0x7f060286;
+ public static final int summary_sort_tiles=0x7f06016a;
+ public static final int summary_studyon=0x7f060299;
+ public static final int summary_thumbsize=0x7f060289;
/**
*/
- public static final int summary_wait_guest=0x7f06020f;
+ public static final int summary_wait_guest=0x7f060212;
/**
*/
- public static final int summary_wait_host=0x7f06020e;
+ public static final int summary_wait_host=0x7f060211;
/** Used to indicate that a preference is not enabled, i.e. not
part of the game and that the user should ignore it.
*/
- public static final int tell_unused=0x7f060135;
- public static final int thumb_off=0x7f060287;
+ public static final int tell_unused=0x7f060138;
+ public static final int thumb_off=0x7f06028a;
/** color of the tiles' background
*/
- public static final int tile_back=0x7f060159;
+ public static final int tile_back=0x7f06015c;
/**
*/
- public static final int tilepick_all=0x7f0601f8;
+ public static final int tilepick_all=0x7f0601fb;
/**
*/
- public static final int tilepick_undo=0x7f0601f7;
+ public static final int tilepick_undo=0x7f0601fa;
/** title of dialog brought up in response to the
board_menu_game_left menu. The dialog lists all tiles
remaining in the pool, i.e. not on the board or in the rack
of the player whose turn it is.
*/
- public static final int tiles_left_title=0x7f0601c4;
+ public static final int tiles_left_title=0x7f0601c7;
/**
############################################################
# :Screens:
@@ -2501,7 +2504,7 @@ public final class R {
window title
*/
- public static final int title_dicts_list=0x7f0600a9;
+ public static final int title_dicts_list=0x7f0600ac;
/**
############################################################
# :Screens:
@@ -2510,20 +2513,20 @@ public final class R {
window title (game name substituted for %1$s)
*/
- public static final int title_game_config_fmt=0x7f0600ba;
+ public static final int title_game_config_fmt=0x7f0600bd;
/**
*/
- public static final int title_gamebt_config_fmt=0x7f0600bc;
+ public static final int title_gamebt_config_fmt=0x7f0600bf;
/** alternate window title used when game is networked
*/
- public static final int title_gamenet_config_fmt=0x7f0600bb;
+ public static final int title_gamenet_config_fmt=0x7f0600be;
/** title for popup list of langugages from which user picks
*/
- public static final int title_langs_list=0x7f0600bd;
+ public static final int title_langs_list=0x7f0600c0;
/**
*/
- public static final int title_lookup=0x7f0601ed;
- public static final int title_lookup_study=0x7f0601ee;
+ public static final int title_lookup=0x7f0601f0;
+ public static final int title_lookup_study=0x7f0601f1;
/**
############################################################
# :Screens:
@@ -2536,45 +2539,46 @@ public final class R {
window title
*/
- public static final int title_prefs=0x7f060128;
+ public static final int title_prefs=0x7f06012b;
+ public static final int title_send_data_sms=0x7f06007e;
/** If this preference is checked, tiles in the rack will be
re-ordered alphabetically whenever tiles are added,
i.e. after ever move.
*/
- public static final int title_sort_tiles=0x7f060166;
- public static final int title_studyon=0x7f060295;
+ public static final int title_sort_tiles=0x7f060169;
+ public static final int title_studyon=0x7f060298;
/** title of dialog allowing user to pick tiles "face up". (This
feature is not yet supported on Android.)
*/
- public static final int title_tile_picker=0x7f0601c3;
+ public static final int title_tile_picker=0x7f0601c6;
/** text of checkbox controlling whether there's a game timer
*/
- public static final int use_timer=0x7f0600cf;
+ public static final int use_timer=0x7f0600d2;
/** Used to separate names of players when listing them on one
line in a game summary. The \u0020 is a space in xml.
*/
- public static final int vs_join=0x7f060122;
+ public static final int vs_join=0x7f060125;
/**
*/
- public static final int warn_nomobile_fmt=0x7f06022b;
+ public static final int warn_nomobile_fmt=0x7f06022e;
/**
*/
- public static final int warn_sms_disabled=0x7f060253;
+ public static final int warn_sms_disabled=0x7f060256;
/**
*/
- public static final int warn_unlimited=0x7f060233;
+ public static final int warn_unlimited=0x7f060236;
/**
*/
- public static final int word_search_hint=0x7f0601f6;
- public static final int xlations_enabled_summary=0x7f0602aa;
- public static final int xlations_enabled_title=0x7f0602a9;
- public static final int xlations_locale=0x7f060086;
+ public static final int word_search_hint=0x7f0601f9;
+ public static final int xlations_enabled_summary=0x7f0602ad;
+ public static final int xlations_enabled_title=0x7f0602ac;
+ public static final int xlations_locale=0x7f060089;
/** Empty in English, this should contain the name of the
translator/creator of the strings.xml file for this
language
*/
- public static final int xlator=0x7f0601e5;
- public static final int xwords_nfc_mime=0x7f06006a;
+ public static final int xlator=0x7f0601e8;
+ public static final int xwords_nfc_mime=0x7f06006b;
}
public static final class style {
public static final int about_items=0x7f080005;
diff --git a/xwords4/android/XWords4/res/raw/changes b/xwords4/android/XWords4/res/raw/changes
index 9ba430c0d..e56982a84 100644
--- a/xwords4/android/XWords4/res/raw/changes
+++ b/xwords4/android/XWords4/res/raw/changes
@@ -9,6 +9,7 @@
New with this release
+ - Display board in upside-down portrait orientation too (where OS version allows)
- Bug fix (obscure): don't mangle unicode model names
diff --git a/xwords4/android/XWords4/res/values/common_rsrc.xml b/xwords4/android/XWords4/res/values/common_rsrc.xml
index b678e082c..58216042d 100644
--- a/xwords4/android/XWords4/res/values/common_rsrc.xml
+++ b/xwords4/android/XWords4/res/values/common_rsrc.xml
@@ -41,6 +41,7 @@
key_dict_host3
key_logging_on
key_show_sms
+ key_send_data_sms
key_init_hintsallowed
key_init_nethintsallowed
key_init_autojuggle
@@ -146,6 +147,8 @@
Enable debug features
Menuitems etc. (release builds
only)
+ Send SMS as data
+ (GSM phones only)
Network stats
Game network stats
diff --git a/xwords4/android/XWords4/res/values/strings.xml b/xwords4/android/XWords4/res/values/strings.xml
index 2427314c5..e65cf0882 100644
--- a/xwords4/android/XWords4/res/values/strings.xml
+++ b/xwords4/android/XWords4/res/values/strings.xml
@@ -886,7 +886,7 @@
Include in game listing
- \u003cNothing\u003E
+ \u003CNothing\u003E
Game language#1
@@ -1200,13 +1200,13 @@
substituted for "%1$s". (The funky \u003c and friends are
encodings for the greater-than and less-than symbols which
are not legal in xml strings.)-->
- \u003ca href=\"%1$s\"\u003ETap
- here\u003c/a\u003E (or tap the full link below, or, if you already
+ \u003Ca href=\"%1$s\"\u003ETap
+ here\u003C/a\u003E (or tap the full link below, or, if you already
have Crosswords installed, open the attachment) to accept my
invitation and join this game.
- \u003cbr \\\u003E
- \u003cbr \\\u003E
+ \u003Cbr \\\u003E
+ \u003Cbr \\\u003E
(full link: %1$s )
@@ -2235,5 +2235,7 @@
Enable local translating
Add option to every screen menu
+
+ SMS Data is only available on GSM phones.
diff --git a/xwords4/android/XWords4/res/xml/xwprefs.xml b/xwords4/android/XWords4/res/xml/xwprefs.xml
index 99422088e..10eb90e02 100644
--- a/xwords4/android/XWords4/res/xml/xwprefs.xml
+++ b/xwords4/android/XWords4/res/xml/xwprefs.xml
@@ -344,6 +344,11 @@
android:title="Show SMS sends, receives"
android:defaultValue="false"
/>
+
MAX_LEN_BINARY ) {
+ len = MAX_LEN_BINARY;
+ }
+ end += len;
+ byte[] part = new byte[4 + len];
+ part[0] = (byte)0; // proto
+ part[1] = (byte)msgID;
+ part[2] = (byte)ii;
+ part[3] = (byte)count;
+ System.arraycopy( msg, start, part, 4, len );
+
+ result[ii] = part;
+ start = end;
+ }
+ return result;
+ }
+
private void receive( SMS_CMD cmd, byte[] data, String phone )
{
DataInputStream dis =
@@ -473,6 +536,19 @@ public class SMSService extends XWService {
}
}
+ private void receiveBuffer( byte[] buffer, String senderPhone )
+ {
+ byte proto = buffer[0];
+ int id = buffer[1];
+ int index = buffer[2];
+ int count = buffer[3];
+ byte[] rest = new byte[buffer.length - 4];
+ System.arraycopy( buffer, 4, rest, 0, rest.length );
+ tryAssemble( senderPhone, id, index, count, rest );
+
+ sendResult( MultiEvent.SMS_RECEIVE_OK );
+ }
+
private void receiveBuffer( String as64, String senderPhone )
{
String[] parts = as64.split( ":" );
@@ -488,6 +564,34 @@ public class SMSService extends XWService {
}
}
+ private void tryAssemble( String senderPhone, int id, int index,
+ int count, byte[] msg )
+ {
+ if ( index == 0 && count == 1 ) {
+ disAssemble( senderPhone, msg );
+ } else {
+ // required? Should always be in main thread.
+ synchronized( s_partialMsgs ) {
+ HashMap perPhone =
+ s_partialMsgs.get( senderPhone );
+ if ( null == perPhone ) {
+ perPhone = new HashMap ();
+ s_partialMsgs.put( senderPhone, perPhone );
+ }
+ MsgStore store = perPhone.get( id );
+ if ( null == store ) {
+ store = new MsgStore( id, count, false );
+ perPhone.put( id, store );
+ }
+
+ if ( store.add( index, msg ).isComplete() ) {
+ disAssemble( senderPhone, store.messageData() );
+ perPhone.remove( id );
+ }
+ }
+ }
+ }
+
private void tryAssemble( String senderPhone, int id, int index,
int count, String msg )
{
@@ -504,18 +608,41 @@ public class SMSService extends XWService {
}
MsgStore store = perPhone.get( id );
if ( null == store ) {
- store = new MsgStore( id, count );
+ store = new MsgStore( id, count, true );
perPhone.put( id, store );
}
if ( store.add( index, msg ).isComplete() ) {
- disAssemble( senderPhone, store.message() );
+ disAssemble( senderPhone, store.messageText() );
perPhone.remove( id );
}
}
}
}
+ private void disAssemble( String senderPhone, byte[] fullMsg )
+ {
+ DataInputStream dis =
+ new DataInputStream( new ByteArrayInputStream(fullMsg) );
+ try {
+ byte proto = dis.readByte();
+ if ( SMS_PROTO_VERSION != proto ) {
+ DbgUtils.logf( "SMSService.disAssemble: bad proto %d; dropping",
+ proto );
+ } else {
+ SMS_CMD cmd = SMS_CMD.values()[dis.readByte()];
+ byte[] rest = new byte[dis.available()];
+ dis.read( rest );
+ receive( cmd, rest, senderPhone );
+ }
+ } catch ( java.io.IOException ioe ) {
+ DbgUtils.loge( ioe );
+ } catch ( ArrayIndexOutOfBoundsException oob ) {
+ // enum this older code doesn't know about; drop it
+ DbgUtils.logf( "disAssemble: dropping message with too-new enum" );
+ }
+ }
+
private void disAssemble( String senderPhone, String fullMsg )
{
byte[] data = XwJNI.base64Decode( fullMsg );
@@ -593,6 +720,39 @@ public class SMSService extends XWService {
return success;
}
+ private boolean sendBuffers( byte[][] fragments, String phone, byte[] data )
+ {
+ boolean success = false;
+ try {
+ SmsManager mgr = SmsManager.getDefault();
+ PendingIntent sent = makeStatusIntent( MSG_SENT );
+ PendingIntent delivery = makeStatusIntent( MSG_DELIVERED );
+ for ( byte[] fragment : fragments ) {
+ mgr.sendDataMessage( phone, null, (short)3344, fragment,
+ sent, delivery );
+ }
+ if ( s_showToasts ) {
+ DbgUtils.showf( this, "sent %dth msg", s_nSent );
+ }
+ success = true;
+ } catch ( IllegalArgumentException iae ) {
+ DbgUtils.logf( "sendBuffers(%s): %s", phone, iae.toString() );
+ } catch ( NullPointerException npe ) {
+ DbgUtils.showf( this, "Switching to regular SMS" );
+ s_asData = new Boolean( false );
+ XWPrefs.setPrefsBoolean( this, R.string.key_send_data_sms,
+ false );
+ success = sendAsText( data, phone );
+ } catch ( Exception ee ) {
+ DbgUtils.loge( ee );
+ }
+
+ ConnStatusHandler.updateStatusOut( this, null,
+ CommsConnType.COMMS_CONN_SMS,
+ success );
+ return success;
+ }
+
private static void fillInviteIntent( Intent intent, String phone,
int gameID, String gameName,
int lang, String dict,
@@ -691,14 +851,39 @@ public class SMSService extends XWService {
@Override
public void onReceive(Context arg0, Intent arg1)
{
- if ( Activity.RESULT_OK == getResultCode() ) {
- DbgUtils.logf( "SUCCESS!!!" );
- } else {
- DbgUtils.logf( "FAILURE!!!" );
- }
+ DbgUtils.logf( "SMS delivery result: %s",
+ Activity.RESULT_OK == getResultCode()
+ ? "SUCCESS" : "FAILURE" );
}
};
registerReceiver( m_receiveReceiver, new IntentFilter(MSG_DELIVERED) );
+
+ m_prefsListener = new OnSharedPreferenceChangeListener() {
+ public void onSharedPreferenceChanged( SharedPreferences sp,
+ String key ) {
+ if ( key.equals( getString( R.string.key_show_sms ) ) ) {
+ s_showToasts = null;
+ } else if ( key.equals( getString( R.string
+ .key_send_data_sms ))) {
+ s_asData = null;
+ }
+ }
+ };
+ SharedPreferences sp
+ = PreferenceManager.getDefaultSharedPreferences( this );
+ sp.registerOnSharedPreferenceChangeListener( m_prefsListener );
+ }
+
+ private boolean sendAsText( byte[] data, String phone )
+ {
+ boolean success = false;
+ try {
+ String[] msgs = breakAndEncode( XwJNI.base64Encode( data ) );
+ success = sendBuffers( msgs, phone );
+ } catch ( java.io.IOException ioe ) {
+ DbgUtils.loge( ioe );
+ }
+ return success;
}
private class SMSMsgSink extends MultiMsgSink {
@@ -729,41 +914,68 @@ public class SMSService extends XWService {
}
private class MsgStore {
- String[] m_msgs;
+ String[] m_msgsText;
+ byte[][] m_msgsData;
int m_msgID;
int m_haveCount;
int m_fullLength;
- public MsgStore( int id, int count )
+ public MsgStore( int id, int count, boolean usingStrings )
{
m_msgID = id;
- m_msgs = new String[count];
+ if ( usingStrings ) {
+ m_msgsText = new String[count];
+ } else {
+ m_msgsData = new byte[count][];
+ }
m_fullLength = 0;
}
public MsgStore add( int index, String msg )
{
- if ( null == m_msgs[index] ) {
+ if ( null == m_msgsText[index] ) {
++m_haveCount;
m_fullLength += msg.length();
}
- m_msgs[index] = msg;
+ m_msgsText[index] = msg;
+ return this;
+ }
+
+ public MsgStore add( int index, byte[] msg )
+ {
+ if ( null == m_msgsData[index] ) {
+ ++m_haveCount;
+ m_fullLength += msg.length;
+ }
+ m_msgsData[index] = msg;
return this;
}
public boolean isComplete()
{
- boolean complete = m_msgs.length == m_haveCount;
+ int count = null != m_msgsText ? m_msgsText.length : m_msgsData.length;
+ boolean complete = count == m_haveCount;
return complete;
}
- public String message()
+ public String messageText()
{
StringBuffer sb = new StringBuffer(m_fullLength);
- for ( String msg : m_msgs ) {
+ for ( String msg : m_msgsText ) {
sb.append( msg );
}
return sb.toString();
}
+
+ public byte[] messageData()
+ {
+ byte[] result = new byte[m_fullLength];
+ int indx = 0;
+ for ( byte[] msg : m_msgsData ) {
+ System.arraycopy( msg, 0, result, indx, msg.length );
+ indx += msg.length;
+ }
+ return result;
+ }
}
}
diff --git a/xwords4/android/XWords4/src/org/eehouse/android/xw4/StudyListActivity.java b/xwords4/android/XWords4/src/org/eehouse/android/xw4/StudyListActivity.java
index 1dd5d7dd0..83fbb9c8b 100644
--- a/xwords4/android/XWords4/src/org/eehouse/android/xw4/StudyListActivity.java
+++ b/xwords4/android/XWords4/src/org/eehouse/android/xw4/StudyListActivity.java
@@ -75,7 +75,7 @@ public class StudyListActivity extends ListActivity {
@Override
protected Dialog onCreateDialog( int id )
{
- Dialog dialog = m_dlgt.createDialog( id );
+ Dialog dialog = m_dlgt.onCreateDialog( id );
if ( null == dialog ) {
dialog = super.onCreateDialog( id );
}
diff --git a/xwords4/android/XWords4/src/org/eehouse/android/xw4/UpdateCheckReceiver.java b/xwords4/android/XWords4/src/org/eehouse/android/xw4/UpdateCheckReceiver.java
index 8eabb20dc..a3a04411e 100644
--- a/xwords4/android/XWords4/src/org/eehouse/android/xw4/UpdateCheckReceiver.java
+++ b/xwords4/android/XWords4/src/org/eehouse/android/xw4/UpdateCheckReceiver.java
@@ -74,6 +74,7 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
private static final String k_PARAMS = "params";
private static final String k_DEVID = "did";
private static final String k_XLATEINFO = "xlatinfo";
+ private static final String k_APPGITREV = "apprev";
@Override
public void onReceive( Context context, Intent intent )
@@ -122,6 +123,13 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
JSONObject params = new JSONObject();
PackageManager pm = context.getPackageManager();
String packageName = context.getPackageName();
+ int versionCode;
+ try {
+ versionCode = pm.getPackageInfo( packageName, 0 ).versionCode;
+ } catch ( PackageManager.NameNotFoundException nnfe ) {
+ DbgUtils.loge( nnfe );
+ versionCode = 0;
+ }
// App update
if ( Utils.isGooglePlayApp( context ) ) {
@@ -130,8 +138,6 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
String installer = pm.getInstallerPackageName( packageName );
try {
- int versionCode = pm.getPackageInfo( packageName, 0 ).versionCode;
-
JSONObject appParams = new JSONObject();
appParams.put( k_NAME, packageName );
@@ -143,8 +149,6 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
}
params.put( k_APP, appParams );
params.put( k_DEVID, XWPrefs.getDevID( context ) );
- } catch ( PackageManager.NameNotFoundException nnfe ) {
- DbgUtils.loge( nnfe );
} catch ( org.json.JSONException jse ) {
DbgUtils.loge( jse );
}
@@ -166,7 +170,7 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
}
// Xlations update
- JSONObject xlationUpdate = LocUtils.makeForXlationUpdate( context );
+ JSONArray xlationUpdate = LocUtils.makeForXlationUpdate( context );
if ( null != xlationUpdate ) {
try {
params.put( k_XLATEINFO, xlationUpdate );
@@ -176,8 +180,16 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
}
if ( 0 < params.length() ) {
- new UpdateQueryTask( context, params, fromUI, pm,
- packageName, dals ).execute();
+ try {
+ params.put( k_APPGITREV, BuildConstants.GIT_HASH );
+ params.put( k_NAME, packageName );
+ params.put( k_AVERS, versionCode );
+ DbgUtils.logf( "current update: %s", params.toString() );
+ new UpdateQueryTask( context, params, fromUI, pm,
+ packageName, dals ).execute();
+ } catch ( org.json.JSONException jse ) {
+ DbgUtils.loge( jse );
+ }
}
}
@@ -387,8 +399,8 @@ public class UpdateCheckReceiver extends BroadcastReceiver {
// translations info
if ( jobj.has( k_XLATEINFO ) ) {
- JSONObject data = jobj.getJSONObject( k_XLATEINFO );
- int nAdded = LocUtils.addXlation( m_context, data );
+ JSONArray data = jobj.getJSONArray( k_XLATEINFO );
+ int nAdded = LocUtils.addXlations( m_context, data );
if ( 0 < nAdded ) {
gotOne = true;
String msg = LocUtils.getString( m_context, R.string
diff --git a/xwords4/android/XWords4/src/org/eehouse/android/xw4/Utils.java b/xwords4/android/XWords4/src/org/eehouse/android/xw4/Utils.java
index 537b6c9d6..e99a5fa77 100644
--- a/xwords4/android/XWords4/src/org/eehouse/android/xw4/Utils.java
+++ b/xwords4/android/XWords4/src/org/eehouse/android/xw4/Utils.java
@@ -97,19 +97,39 @@ public class Utils {
return s_isFirstBootThisVersion;
}
- // Does the device have ability to send SMS -- e.g. is it a phone
- // and not a Kindle Fire. Not related to XWApp.SMSSUPPORTED
+ public static boolean isGSMPhone( Context context )
+ {
+ boolean result = false;
+ TelephonyManager tm = (TelephonyManager)
+ context.getSystemService( Context.TELEPHONY_SERVICE );
+ if ( null != tm ) {
+ result = TelephonyManager.PHONE_TYPE_GSM == tm.getPhoneType();
+ }
+ return result;
+ }
+
+ // Does the device have ability to send SMS -- e.g. is it a phone and not
+ // a Kindle Fire. Not related to XWApp.SMSSUPPORTED. Note that as a
+ // temporary workaround for KitKat having broken use of non-data messages,
+ // we only support SMS on kitkat if data messages have been turned on (and
+ // that's not allowed except on GSM phones.)
public static boolean deviceSupportsSMS( Context context )
{
if ( null == s_deviceSupportSMS ) {
boolean doesSMS = false;
- // TEMPORARY: disable SMS on KITKAT
- if ( 19 > Integer.valueOf( android.os.Build.VERSION.SDK ) ) {
+ // TEMPORARY: disable SMS on KITKAT UNLESS use-text turned on
+ boolean preKitkat = 19 > Integer.valueOf( android.os.Build.VERSION.SDK);
+ boolean usingData =
+ XWPrefs.getPrefsBoolean( context, R.string.key_send_data_sms,
+ false );
+ if ( preKitkat || usingData ) {
TelephonyManager tm = (TelephonyManager)
context.getSystemService(Context.TELEPHONY_SERVICE);
if ( null != tm ) {
int type = tm.getPhoneType();
- doesSMS = TelephonyManager.PHONE_TYPE_NONE != type;
+ doesSMS = (usingData && !preKitkat)
+ ? TelephonyManager.PHONE_TYPE_GSM == type
+ : TelephonyManager.PHONE_TYPE_NONE != type;
}
}
s_deviceSupportSMS = new Boolean( doesSMS );
diff --git a/xwords4/android/XWords4/src/org/eehouse/android/xw4/XWPrefs.java b/xwords4/android/XWords4/src/org/eehouse/android/xw4/XWPrefs.java
index f61fb21c5..aa00f0e32 100644
--- a/xwords4/android/XWords4/src/org/eehouse/android/xw4/XWPrefs.java
+++ b/xwords4/android/XWords4/src/org/eehouse/android/xw4/XWPrefs.java
@@ -430,7 +430,7 @@ public class XWPrefs {
setPrefsString( context, keyID, TextUtils.join( "\n", value ) );
}
- public static String getLocale( Context context )
+ public static String getFakeLocale( Context context )
{
return getPrefsString( context, R.string.key_xlations_locale );
}
diff --git a/xwords4/android/XWords4/src/org/eehouse/android/xw4/loc/LocUtils.java b/xwords4/android/XWords4/src/org/eehouse/android/xw4/loc/LocUtils.java
index 8b6c793fc..89f75fae8 100644
--- a/xwords4/android/XWords4/src/org/eehouse/android/xw4/loc/LocUtils.java
+++ b/xwords4/android/XWords4/src/org/eehouse/android/xw4/loc/LocUtils.java
@@ -36,12 +36,18 @@ import android.widget.Button;
import android.widget.CheckBox;
import android.widget.Spinner;
import android.widget.TextView;
+import android.preference.Preference;
+import android.preference.PreferenceGroup;
+import android.preference.PreferenceActivity;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
+import java.util.Locale;
import java.util.Map;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
import org.json.JSONArray;
import org.json.JSONObject;
@@ -55,12 +61,8 @@ import org.eehouse.android.xw4.Utils;
import org.eehouse.android.xw4.XWPrefs;
public class LocUtils {
- // Keep this in sync with gen_loc_ids.py and what's used in the menu.xml
- // files to mark me-localized strings.
private static final int FMT_LEN = 4;
private static final String k_LOCALE = "locale";
- private static final String k_XLATPROTO = "proto";
- private static final int XLATE_CUR_VERSION = 1;
private static final String k_XLATEVERS = "xlatevers";
private static Map s_xlationsLocal = null;
@@ -121,6 +123,11 @@ public class LocUtils {
xlateView( activity, Utils.getContentView( activity ) );
}
+ public static void xlatePreferences( PreferenceActivity activity )
+ {
+ xlatePreferences( activity, activity.getPreferenceScreen(), 0 );
+ }
+
public static void xlateView( Context context, View view )
{
DbgUtils.logf( "xlateView() top level" );
@@ -132,6 +139,15 @@ public class LocUtils {
xlateMenu( activity, menu, 0 );
}
+ private static String xlateString( Context context, CharSequence str )
+ {
+ String result = null;
+ if ( null != str ) {
+ result = xlateString( context, str.toString() );
+ }
+ return result;
+ }
+
public static String xlateString( Context context, String str )
{
if ( LocIDs.getS_MAP( context ).containsKey( str ) ) {
@@ -225,55 +241,77 @@ public class LocUtils {
public static void saveLocalData( Context context )
{
- DBUtils.saveXlations( context, s_curLocale, s_xlationsLocal, false );
+ DBUtils.saveXlations( context, getCurLocale( context ),
+ s_xlationsLocal, false );
}
- public static JSONObject makeForXlationUpdate( Context context )
+ public static JSONArray makeForXlationUpdate( Context context )
{
- JSONObject result = null;
- if ( null != s_curLocale && 0 < s_curLocale.length() ) {
- try {
- String version = DBUtils.getStringFor( context, k_XLATEVERS, "0" );
- result = new JSONObject()
- .put( k_XLATPROTO, XLATE_CUR_VERSION )
- .put( k_LOCALE, s_curLocale )
- .put( k_XLATEVERS, version );
- } catch ( org.json.JSONException jse ) {
- DbgUtils.loge( jse );
- }
+ String locale = getCurLocale( context );
+ String fake = XWPrefs.getFakeLocale( context );
+ JSONArray result = new JSONArray()
+ .put( entryForLocale( context, locale ) );
+ if ( null != fake && 0 < fake.length() && ! fake.equals(locale) ) {
+ result.put( entryForLocale( context, fake ) );
}
return result;
}
+ private static JSONObject entryForLocale( Context context, String locale )
+ {
+ JSONObject result = null;
+ try {
+ String version =
+ DBUtils.getStringFor( context, localeKey(locale), "0" );
+ result = new JSONObject()
+ .put( k_LOCALE, locale )
+ .put( k_XLATEVERS, version );
+ } catch ( org.json.JSONException jse ) {
+ DbgUtils.loge( jse );
+ }
+ return result;
+ }
+
+ private static String localeKey( String locale )
+ {
+ return String.format( "%s:%s", k_XLATEVERS, locale );
+ }
+
private static final String k_OLD = "old";
private static final String k_NEW = "new";
private static final String k_PAIRS = "pairs";
- public static int addXlation( Context context, JSONObject data )
+ public static int addXlations( Context context, JSONArray data )
{
int nAdded = 0;
try {
- int newVersion = data.getInt( k_NEW );
- JSONArray pairs = data.getJSONArray( k_PAIRS );
- DbgUtils.logf( "got pairs of len %d, version %d", pairs.length(),
- newVersion );
+ int nLocales = data.length();
+ for ( int ii = 0; ii < nLocales; ++ii ) {
+ JSONObject entry = data.getJSONObject( ii );
+ String locale = entry.getString( k_LOCALE );
+ String newVersion = entry.getString( k_NEW );
+ JSONArray pairs = entry.getJSONArray( k_PAIRS );
+ DbgUtils.logf( "addXlations: locale %s: got pairs of len %d, version %s", locale,
+ pairs.length(), newVersion );
- int len = pairs.length();
- Map newXlations = new HashMap( len );
- for ( int ii = 0; ii < len; ++ii ) {
- JSONObject pair = pairs.getJSONObject( ii );
- String key = pair.getString( "en" );
- String txt = pair.getString( "loc" );
- newXlations.put( key, txt );
+ int len = pairs.length();
+ Map newXlations = new HashMap( len );
+ for ( int jj = 0; jj < len; ++jj ) {
+ JSONObject pair = pairs.getJSONObject( jj );
+ int id = pair.getInt( "id" );
+ String key = context.getString( id );
+ Assert.assertNotNull( key );
+ String txt = pair.getString( "loc" );
+ txt = replaceEscaped( txt );
+ newXlations.put( key, txt );
+ }
+
+ DBUtils.saveXlations( context, locale, newXlations, true );
+ DBUtils.setStringFor( context, localeKey(locale), newVersion );
+ nAdded += len;
}
-
- DBUtils.saveXlations( context, s_curLocale, newXlations, true );
- DBUtils.setStringFor( context, k_XLATEVERS,
- String.format( "%d", newVersion ) );
-
s_xlationsBlessed = null;
loadXlations( context );
- nAdded = len;
} catch ( org.json.JSONException jse ) {
DbgUtils.loge( jse );
}
@@ -334,13 +372,23 @@ public class LocUtils {
}
}
- private static void loadXlations( Context context )
+ private static String getCurLocale( Context context )
{
if ( null == s_curLocale ) {
- s_curLocale = XWPrefs.getLocale( context );
+ String locale = XWPrefs.getFakeLocale( context );
+ if ( null == locale || 0 == locale.length() ) {
+ locale = Locale.getDefault().toString();
+ }
+ s_curLocale = locale;
}
+ return s_curLocale;
+ }
+
+ private static void loadXlations( Context context )
+ {
if ( null == s_xlationsLocal || null == s_xlationsBlessed ) {
- Object[] asObjs = DBUtils.getXlations( context, s_curLocale );
+ Object[] asObjs = DBUtils.getXlations( context,
+ getCurLocale( context ) );
s_xlationsLocal = (Map)asObjs[0];
s_xlationsBlessed = (Map)asObjs[1];
DbgUtils.logf( "loadXlations: got %d local strings, %d blessed strings",
@@ -367,16 +415,6 @@ public class LocUtils {
return s_idsToKeys.get( id );
}
- private static boolean isEnabled( Context context )
- {
- if ( null == s_enabled ) {
- s_curLocale = XWPrefs.getLocale( context );
- s_enabled = new Boolean( null != s_curLocale &&
- 0 < s_curLocale.length() );
- }
- return s_enabled;
- }
-
private static void xlateView( Context context, View view, int depth )
{
// DbgUtils.logf( "xlateView(depth=%d, view=%s, canRecurse=%b)", depth,
@@ -388,11 +426,6 @@ public class LocUtils {
} else if ( view instanceof TextView ) {
TextView tv = (TextView)view;
tv.setText( xlateString( context, tv.getText().toString() ) );
- // } else if ( view instanceof CheckBox ) {
- // CheckBox box = (CheckBox)view;
- // String str = box.getText().toString();
- // str = xlateString( context, str );
- // box.setText( str );
} else if ( view instanceof Spinner ) {
Spinner sp = (Spinner)view;
CharSequence prompt = sp.getPrompt();
@@ -415,23 +448,80 @@ public class LocUtils {
}
}
+ public static void xlatePreferences( Context context, Preference pref,
+ int depth )
+ {
+ // DbgUtils.logf( "xlatePreferences(depth=%d, view=%s, canRecurse=%b)", depth,
+ // pref.getClass().getName(), pref instanceof PreferenceGroup );
+
+ String str = xlateString( context, pref.getSummary() );
+ if ( null != str ) {
+ pref.setSummary( str );
+ }
+ str = xlateString( context, pref.getTitle() );
+ if ( null != str ) {
+ pref.setTitle( str );
+ }
+
+ if ( pref instanceof PreferenceGroup ) {
+ PreferenceGroup group = (PreferenceGroup)pref;
+ int count = group.getPreferenceCount();
+ for ( int ii = 0; ii < count; ++ii ) {
+ xlatePreferences( context, group.getPreference(ii), 1 + depth );
+ }
+ }
+ }
+
// This is for testing, but the ability to pull the formatters will be
// critical for validating local transations of strings containing
// formatters.
private static String toUpperCase( String str )
{
- String[] parts = str.split( "%[\\d]\\$[ds]" );
- StringBuilder sb = new StringBuilder();
- int offset = 0;
- for ( String part : parts ) {
- sb.append( part.toUpperCase() );
- offset += part.length();
- if ( offset < str.length() ) {
- sb.append( str.substring( offset, offset + FMT_LEN ) );
- offset += FMT_LEN;
+ String result = null;
+ if ( UPPER_CASE ) {
+ String[] parts = str.split( "%[\\d]\\$[ds]" );
+ StringBuilder sb = new StringBuilder();
+ int offset = 0;
+ for ( String part : parts ) {
+ sb.append( part.toUpperCase() );
+ offset += part.length();
+ if ( offset < str.length() ) {
+ sb.append( str.substring( offset, offset + FMT_LEN ) );
+ offset += FMT_LEN;
+ }
}
+ result = sb.toString();
}
- return sb.toString();
+ return result;
+ }
+
+ private static Pattern s_patUnicode = Pattern.compile("(\\\\[Uu][0-9a-fA-F]{4})");
+ private static Pattern s_patCr = Pattern.compile("\\\\n");
+
+ private static String replaceEscaped( String txt )
+ {
+ // String orig = txt;
+
+ // Swap unicode escapes for real chars
+ Matcher matcher = s_patUnicode.matcher( txt );
+ StringBuffer sb = new StringBuffer();
+ while ( matcher.find() ) {
+ int start = matcher.start();
+ int end = matcher.end();
+ String match = txt.substring( start, end );
+ char ch = (char)Integer.parseInt( match.substring(2), 16 );
+ matcher.appendReplacement( sb, String.valueOf(ch) );
+ }
+ matcher.appendTail(sb);
+ txt = sb.toString();
+
+ // Swap in real carriage returns
+ txt = s_patCr.matcher( txt ).replaceAll( "\n" );
+
+ // if ( ! orig.equals( txt ) ) {
+ // DbgUtils.logf( "replaceEscaped: <<%s>> -> <<%s>>", orig, txt );
+ // }
+ return txt;
}
public static AlertDialog.Builder makeAlertBuilder( Context context )
diff --git a/xwords4/android/scripts/arelease.sh b/xwords4/android/scripts/arelease.sh
index 642e33b87..cc8ff9e9d 100755
--- a/xwords4/android/scripts/arelease.sh
+++ b/xwords4/android/scripts/arelease.sh
@@ -77,7 +77,7 @@ for PACK_UNSIGNED in $FILES; do
if [ -n "$XW_RELEASE_SCP_DEST" ]; then
echo "running scp ${TARGET} $XW_RELEASE_SCP_DEST"
- scp "${TARGET}" "$XW_RELEASE_SCP_DEST"
+ scp "${TARGET}" "$XW_RELEASE_SCP_DEST" || true
else
echo "XW_RELEASE_SCP_DEST not set; you're on your own"
fi
diff --git a/xwords4/android/scripts/genvers.sh b/xwords4/android/scripts/genvers.sh
index 29c92f738..c6d33cf04 100755
--- a/xwords4/android/scripts/genvers.sh
+++ b/xwords4/android/scripts/genvers.sh
@@ -2,6 +2,13 @@
set -e -u
+usage() {
+ echo "usage: $0 "
+ exit 1
+}
+
+[ $# -eq 4 ] || usage
+
VARIANT=$1
CLIENT_VERS_RELAY=$2
CHAT_SUPPORTED=$3
@@ -12,6 +19,11 @@ cd $(dirname $0)
cd ../
GITVERSION=$(../scripts/gitversion.sh)
+if git rev-parse $GITVERSION 2>/dev/null 1>/dev/null; then
+ GIT_HASH=$(git rev-parse $GITVERSION)
+else
+ GIT_HASH=unknown
+fi
# TODO: deal with case where there's no hash available -- exported
# code maybe? Better: gitversion.sh does that.
@@ -34,6 +46,7 @@ cat < ${BUILD_DIR}/src/org/eehouse/android/${VARIANT}/BuildConstants.java
package org.eehouse.android.${VARIANT};
class BuildConstants {
public static final String GIT_REV = "$SHORTVERS";
+ public static final String GIT_HASH = "$GIT_HASH";
public static final short CLIENT_VERS_RELAY = $CLIENT_VERS_RELAY;
public static final boolean CHAT_SUPPORTED = $CHAT_SUPPORTED;
public static final boolean THUMBNAIL_SUPPORTED = $THUMBNAIL_SUPPORTED;
diff --git a/xwords4/relay/Makefile b/xwords4/relay/Makefile
index b85aeb906..b762735f0 100644
--- a/xwords4/relay/Makefile
+++ b/xwords4/relay/Makefile
@@ -38,6 +38,7 @@ SRC = \
udpager.cpp \
udpqueue.cpp \
xwrelay.cpp \
+ querybld.cpp \
# STATIC ?= -static
GITINFO = gitversion.txt
diff --git a/xwords4/relay/dbmgr.cpp b/xwords4/relay/dbmgr.cpp
index 212454a65..179b01dad 100644
--- a/xwords4/relay/dbmgr.cpp
+++ b/xwords4/relay/dbmgr.cpp
@@ -43,11 +43,8 @@
static DBMgr* s_instance = NULL;
-#define DELIM "\1"
#define MAX_NUM_PLAYERS 4
-static void formatParams( char* paramValues[], int nParams, const char* fmt,
- char* buf, int bufLen, ... );
static int here_less_seed( const char* seeds, int perDeviceSum,
unsigned short seed );
static void destr_function( void* conn );
@@ -105,20 +102,21 @@ DBMgr::AddNew( const char* cookie, const char* connName, CookieID cid,
if ( !cookie ) cookie = "";
if ( !connName ) connName = "";
- const char* command = "INSERT INTO " GAMES_TABLE
- " (cid, room, connName, nTotal, lang, pub)"
- " VALUES( $1, $2, $3, $4, $5, $6 )";
- int nParams = 6;
- char* paramValues[nParams];
- char buf[512];
- formatParams( paramValues, nParams,
- "%d"DELIM"%s"DELIM"%s"DELIM"%d"DELIM"%d"DELIM"%s",
- buf, sizeof(buf), cid, cookie, connName, nPlayersT,
- langCode, isPublic?"TRUE":"FALSE" );
+ QueryBuilder qb;
+ qb.appendQueryf( "INSERT INTO " GAMES_TABLE
+ " (cid, room, connName, nTotal, lang, pub)"
+ " VALUES( $$, $$, $$, $$, $$, $$ )" )
+ .appendParam(cid)
+ .appendParam(cookie)
+ .appendParam(connName)
+ .appendParam(nPlayersT)
+ .appendParam(langCode)
+ .appendParam(isPublic?"TRUE":"FALSE" )
+ .finish();
- PGresult* result = PQexecParams( getThreadConn(), command,
- nParams, NULL,
- paramValues,
+ PGresult* result = PQexecParams( getThreadConn(), qb.c_str(),
+ qb.paramCount(), NULL,
+ qb.paramValues(),
NULL, NULL, 0 );
if ( PGRES_COMMAND_OK != PQresultStatus(result) ) {
logf( XW_LOGERROR, "PQexec=>%s;%s", PQresStatus(PQresultStatus(result)),
@@ -271,28 +269,27 @@ DBMgr::SeenSeed( const char* cookie, unsigned short seed,
char* connNameBuf, int bufLen, int* nPlayersHP,
CookieID* cid )
{
- int nParams = 5;
- char* paramValues[nParams];
- char buf[512];
- formatParams( paramValues, nParams,
- "%s"DELIM"%d"DELIM"%d"DELIM"%d"DELIM"%s", buf, sizeof(buf),
- cookie, langCode, nPlayersT, seed,
- wantsPublic?"TRUE":"FALSE" );
+ QueryBuilder qb;
+ qb.appendQueryf( "SELECT cid, connName, seeds, sum_array(nPerDevice) FROM "
+ GAMES_TABLE
+ " WHERE NOT dead"
+ " AND room ILIKE $$"
+ " AND lang = $$"
+ " AND nTotal = $$"
+ " AND $$ = ANY(seeds)"
+ " AND $$ = pub"
+ " ORDER BY ctime DESC"
+ " LIMIT 1")
+ .appendParam(cookie)
+ .appendParam(langCode)
+ .appendParam(nPlayersT)
+ .appendParam(seed)
+ .appendParam(wantsPublic?"TRUE":"FALSE" )
+ .finish();
- const char* cmd = "SELECT cid, connName, seeds, sum_array(nPerDevice) FROM "
- GAMES_TABLE
- " WHERE NOT dead"
- " AND room ILIKE $1"
- " AND lang = $2"
- " AND nTotal = $3"
- " AND $4 = ANY(seeds)"
- " AND $5 = pub"
- " ORDER BY ctime DESC"
- " LIMIT 1";
-
- PGresult* result = PQexecParams( getThreadConn(), cmd,
- nParams, NULL,
- paramValues,
+ PGresult* result = PQexecParams( getThreadConn(), qb.c_str(),
+ qb.paramCount(), NULL,
+ qb.paramValues(),
NULL, NULL, 0 );
bool found = 1 == PQntuples( result );
if ( found ) {
@@ -312,31 +309,28 @@ DBMgr::FindOpen( const char* cookie, int lang, int nPlayersT, int nPlayersH,
bool wantsPublic, char* connNameBuf, int bufLen,
int* nPlayersHP )
{
- CookieID cid = 0;
+ QueryBuilder qb;
+ qb.appendQueryf("SELECT cid, connName, sum_array(nPerDevice) FROM "
+ GAMES_TABLE
+ " WHERE NOT dead"
+ " AND room ILIKE $$"
+ " AND lang = $$"
+ " AND nTotal = $$"
+ " AND $$ <= nTotal-sum_array(nPerDevice)"
+ " AND $$ = pub"
+ " LIMIT 1")
+ .appendParam(cookie)
+ .appendParam(lang)
+ .appendParam(nPlayersT)
+ .appendParam(nPlayersH)
+ .appendParam(wantsPublic?"TRUE":"FALSE" )
+ .finish();
- int nParams = 5;
- char* paramValues[nParams];
- char buf[512];
- formatParams( paramValues, nParams,
- "%s"DELIM"%d"DELIM"%d"DELIM"%d"DELIM"%s", buf, sizeof(buf),
- cookie, lang, nPlayersT, nPlayersH, wantsPublic?"TRUE":"FALSE" );
-
- /* NOTE: ILIKE, for case-insensitive comparison, is a postgres extension
- to SQL. */
- const char* cmd = "SELECT cid, connName, sum_array(nPerDevice) FROM "
- GAMES_TABLE
- " WHERE NOT dead"
- " AND room ILIKE $1"
- " AND lang = $2"
- " AND nTotal = $3"
- " AND $4 <= nTotal-sum_array(nPerDevice)"
- " AND $5 = pub"
- " LIMIT 1";
-
- PGresult* result = PQexecParams( getThreadConn(), cmd,
- nParams, NULL,
- paramValues,
+ PGresult* result = PQexecParams( getThreadConn(), qb.c_str(),
+ qb.paramCount(), NULL,
+ qb.paramValues(),
NULL, NULL, 0 );
+ CookieID cid = 0;
if ( 1 == PQntuples( result ) ) {
cid = atoi( PQgetvalue( result, 0, 0 ) );
snprintf( connNameBuf, bufLen, "%s", PQgetvalue( result, 0, 1 ) );
@@ -396,14 +390,21 @@ DBMgr::RegisterDevice( const DevID* host, int clientVersion,
devID = (DevIDRelay)random();
} while ( DEVID_NONE == devID );
- StrWPF query;
- query.catf( "INSERT INTO " DEVICES_TABLE " (id, devTypes[1],"
- " devids[1], clntVers, versdesc, model, osvers)"
- " VALUES( %d, %d, '%s', %d, '%s', '%s', '%s' )",
- devID, host->m_devIDType, devidStr, clientVersion,
- desc, model, osVers );
- logf( XW_LOGINFO, "%s: %s", __func__, query.c_str() );
- success = execSql( query );
+ QueryBuilder qb;
+ qb.appendQueryf( "INSERT INTO " DEVICES_TABLE " (id, devTypes[1],"
+ " devids[1], clntVers, versdesc, model, osvers)"
+ " VALUES($$, $$, $$, $$, $$, $$, $$)" );
+
+ qb.appendParam( devID )
+ .appendParam( host->m_devIDType )
+ .appendParam( devidStr )
+ .appendParam( clientVersion )
+ .appendParam( desc )
+ .appendParam( model )
+ .appendParam( osVers )
+ .finish();
+
+ success = execParams( qb );
}
}
return devID;
@@ -420,15 +421,17 @@ DBMgr::ReregisterDevice( DevIDRelay relayID, const DevID* host,
const char* const desc, int clientVersion,
const char* const model, const char* const osVers )
{
- // First update the existing
- StrWPF query;
- query.catf( "UPDATE " DEVICES_TABLE " SET "
- "devTypes = array_prepend( %d, devTypes), "
- "devids = array_prepend('%s', devids), ",
- host->m_devIDType, host->m_devIDString.c_str() );
+ QueryBuilder qb;
+ qb.appendQueryf( "UPDATE " DEVICES_TABLE " SET "
+ "devTypes = array_prepend($$, devTypes), "
+ "devids = array_prepend($$, devids), " )
- formatUpdate( query, true, desc, clientVersion, model, osVers, relayID );
- execSql( query );
+ .appendParam( host->m_devIDType )
+ .appendParam( host->m_devIDString.c_str() );
+
+ formatUpdate( qb, true, desc, clientVersion, model, osVers, relayID );
+ qb.finish();
+ execParams( qb );
}
// Return true if the relayID exists in the DB already
@@ -445,10 +448,11 @@ DBMgr::UpdateDevice( DevIDRelay relayID, const char* const desc,
}
if ( exists ) {
- StrWPF query;
- query.catf( "UPDATE " DEVICES_TABLE " SET " );
- formatUpdate( query, false, desc, clientVersion, model, osVers, relayID );
- execSql( query );
+ QueryBuilder qb;
+ qb.appendQueryf( "UPDATE " DEVICES_TABLE " SET " );
+ formatUpdate( qb, false, desc, clientVersion, model, osVers, relayID );
+ qb.finish();
+ execParams( qb );
}
return exists;
}
@@ -460,26 +464,33 @@ DBMgr::UpdateDevice( DevIDRelay relayID )
}
void
-DBMgr::formatUpdate( StrWPF& query, bool append, const char* const desc,
+DBMgr::formatUpdate( QueryBuilder& qb,
+ bool append, const char* const desc,
int clientVersion, const char* const model,
const char* const osVers, DevIDRelay relayID )
{
if ( append ) {
- query.catf( "mtimes=array_prepend('now', mtimes)" ); // FIXME: too many
+ qb.appendQueryf( "mtimes=array_prepend('now', mtimes)" ); // FIXME: too many
} else {
- query.catf( "mtimes[1]='now'" );
+ qb.appendQueryf( "mtimes[1]='now'" );
}
if ( NULL != desc && '\0' != desc[0] ) {
- query.catf( ", clntVers=%d, versDesc='%s'", clientVersion, desc );
+ qb.appendQueryf( ", clntVers=$$" )
+ .appendParam( clientVersion )
+ .appendQueryf( ", versDesc=$$" )
+ .appendParam( desc );
}
if ( NULL != model && '\0' != model[0] ) {
- query.catf( ", model='%s'", model );
+ qb.appendQueryf( ", model=$$" )
+ .appendParam( model );
}
if ( NULL != osVers && '\0' != osVers[0] ) {
- query.catf( ", osvers='%s'", osVers );
+ qb.appendQueryf( ", osvers=$$" )
+ .appendParam( osVers );
}
- query.catf( " WHERE id = %d", relayID );
+ qb.appendQueryf( " WHERE id = $$" )
+ .appendParam( relayID );
}
HostID
@@ -831,6 +842,22 @@ DBMgr::execSql( const char* const query )
return ok;
}
+bool
+DBMgr::execParams( QueryBuilder& qb )
+{
+ PGresult* result = PQexecParams( getThreadConn(), qb.c_str(),
+ qb.paramCount(), NULL,
+ qb.paramValues(),
+ NULL, NULL, 0 );
+ bool success = PGRES_COMMAND_OK == PQresultStatus( result );
+ if ( !success ) {
+ logf( XW_LOGERROR, "PQexecParams(%s)=>%s;%s", qb.c_str(),
+ PQresStatus(PQresultStatus(result)),
+ PQresultErrorMessage(result) );
+ }
+ return success;
+}
+
void
DBMgr::readArray( const char* const connName, const char* column, int arr[] ) /* len 4 */
{
@@ -1255,31 +1282,6 @@ void DBMgr::clearHasNoMessages( DevIDRelay devid )
assert( !hasNoMessages( devid ) );
}
-static void
-formatParams( char* paramValues[], int nParams, const char* fmt, char* buf,
- int bufLen, ... )
-{
- va_list ap;
- va_start( ap, bufLen );
-
- int len = vsnprintf( buf, bufLen, fmt, ap );
- assert( buf[len] == '\0' );
-
- int pnum;
- char* ptr = buf;
- for ( pnum = 0; pnum < nParams; ++pnum ) {
- paramValues[pnum] = ptr;
- for ( ; *ptr != '\0' && *ptr != DELIM[0]; ++ptr ) {
- // do nothing
- assert( ptr < &buf[bufLen] );
- }
- // we've found an end
- *ptr = '\0';
- ++ptr;
- }
- va_end(ap);
-}
-
static int
here_less_seed( const char* seeds, int sumPerDevice, unsigned short seed )
{
diff --git a/xwords4/relay/dbmgr.h b/xwords4/relay/dbmgr.h
index 6e5bcbf0e..e0fafcaad 100644
--- a/xwords4/relay/dbmgr.h
+++ b/xwords4/relay/dbmgr.h
@@ -32,6 +32,7 @@
#include "xwrelay_priv.h"
#include "devid.h"
#include "strwpf.h"
+#include "querybld.h"
using namespace std;
@@ -149,6 +150,7 @@ class DBMgr {
DBMgr();
bool execSql( const string& query );
bool execSql( const char* const query ); /* no-results query */
+ bool execParams( QueryBuilder& qb );
void readArray( const char* const connName, const char* column, int arr[] );
DevIDRelay getDevID( const char* connName, int hid );
DevIDRelay getDevID( const DevID* devID );
@@ -160,7 +162,7 @@ class DBMgr {
bool nullConnnameOK );
int CountStoredMessages( const char* const connName, int hid );
bool UpdateDevice( DevIDRelay relayID );
- void formatUpdate( StrWPF& query, bool append, const char* const desc,
+ void formatUpdate( QueryBuilder& qb, bool append, const char* const desc,
int clientVersion, const char* const model,
const char* const osVers, DevIDRelay relayID );
diff --git a/xwords4/relay/querybld.cpp b/xwords4/relay/querybld.cpp
new file mode 100644
index 000000000..6334d1f9f
--- /dev/null
+++ b/xwords4/relay/querybld.cpp
@@ -0,0 +1,78 @@
+/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
+/*
+ * Copyright 2014 by Eric House (xwords@eehouse.org). All rights reserved.
+ *
+ * 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 2
+ * of the License, or (at your option.
+ *
+ * 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, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+#include
+
+#include "querybld.h"
+#include "xwrelay_priv.h"
+
+QueryBuilder&
+QueryBuilder::appendQueryf( const char* fmt, ... )
+{
+ bool done;
+ do {
+ va_list ap;
+ va_start( ap, fmt );
+ done = m_query.catf( fmt, ap );
+ va_end( ap );
+ } while ( !done );
+ return *this;
+}
+
+QueryBuilder&
+QueryBuilder::appendParam( const char* value )
+{
+ m_paramIndices.push_back( m_paramBuf.size() );
+ m_paramBuf.catf( "%s%c", value, '\0' );
+ return *this;
+}
+
+QueryBuilder&
+QueryBuilder::appendParam( int value )
+{
+ m_paramIndices.push_back( m_paramBuf.size() );
+ m_paramBuf.catf( "%d%c", value, '\0' );
+ return *this;
+}
+
+/* When done adding params, some of which contain $$, turn these into an order
+ * progression of $1, $2 .. $9. Note assumption that we don't go above 9 since
+ */
+void
+QueryBuilder::finish()
+{
+ assert( 0 == m_paramValues.size() );
+
+ size_t ii;
+ const char* base = m_paramBuf.c_str();
+ for ( ii = 0; ii < m_paramIndices.size(); ++ii ) {
+ const char* ptr = m_paramIndices[ii] + base;
+ m_paramValues.push_back( ptr );
+ }
+
+ for ( size_t count = 0; ; ++count ) {
+ const char* str = m_query.c_str();
+ const char* ptr = strstr( str, "$$" );
+ if ( !ptr ) {
+ assert( count == m_paramIndices.size() );
+ break;
+ }
+ assert( count < 9 );
+ m_query[1 + ptr - str] = '1' + count;
+ }
+}
diff --git a/xwords4/relay/querybld.h b/xwords4/relay/querybld.h
new file mode 100644
index 000000000..1def83271
--- /dev/null
+++ b/xwords4/relay/querybld.h
@@ -0,0 +1,47 @@
+/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
+/*
+ * Copyright 2014 by Eric House (xwords@eehouse.org). All rights reserved.
+ *
+ * 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 2
+ * of the License, or (at your option.
+ *
+ * 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, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+
+#ifndef _QUERYBLD_H_
+#define _QUERYBLD_H_
+
+#include
+
+#include "strwpf.h"
+
+using namespace std;
+
+class QueryBuilder {
+
+ public:
+ QueryBuilder& appendQueryf( const char* fmt, ... );
+ QueryBuilder& appendParam( const char* value );
+ QueryBuilder& appendParam( int value );
+ void finish();
+ int paramCount() const { return m_paramValues.size(); }
+ const char* const* paramValues() const { return &m_paramValues[0]; }
+ const char* const c_str() const { return m_query.c_str(); }
+
+ private:
+ StrWPF m_query;
+ StrWPF m_paramBuf;
+ vector m_paramIndices;
+ vector m_paramValues;
+};
+
+#endif
diff --git a/xwords4/relay/strwpf.cpp b/xwords4/relay/strwpf.cpp
index 9a8ab0a83..d6085755a 100644
--- a/xwords4/relay/strwpf.cpp
+++ b/xwords4/relay/strwpf.cpp
@@ -27,26 +27,37 @@
/* From stack overflow: snprintf with an expanding buffer.
*/
+bool
+StrWPF::catf( const char* fmt, va_list ap )
+{
+ bool success = false;
+ const int origsiz = size();
+ resize( origsiz + m_addsiz );
+
+ int len = vsnprintf( (char*)c_str() + origsiz, m_addsiz, fmt, ap );
+
+ if ( len >= m_addsiz ) { // needs more space
+ m_addsiz = len + 1;
+ resize( origsiz );
+ } else if ( -1 == len ) {
+ assert(0); // should be impossible
+ } else {
+ resize( origsiz + len );
+ m_addsiz = 100;
+ success = true;
+ }
+
+ return success;
+}
+
void
StrWPF::catf( const char* fmt, ... )
{
- const int origsiz = size();
- int addsiz = 100;
- va_list ap;
- for ( ; ; ) {
- resize( origsiz + addsiz );
-
+ bool done;
+ do {
+ va_list ap;
va_start( ap, fmt );
- int len = vsnprintf( (char *)c_str() + origsiz, addsiz, fmt, ap );
+ done = catf( fmt, ap );
va_end( ap );
-
- if ( len >= addsiz ) { // needs more space
- addsiz = len + 1;
- } else if ( -1 == len ) {
- assert(0); // should be impossible
- } else {
- resize( origsiz + len );
- break;
- }
- }
+ } while ( !done );
}
diff --git a/xwords4/relay/strwpf.h b/xwords4/relay/strwpf.h
index 950a9e817..e3973d2dc 100644
--- a/xwords4/relay/strwpf.h
+++ b/xwords4/relay/strwpf.h
@@ -21,10 +21,16 @@
#define _STRWPF_H_
#include
+#include
-class StrWPF : public std::string {
+class StrWPF : public std::string {
public:
+ StrWPF() : m_addsiz(100){}
+
void catf( const char* fmt, ... );
+ bool catf( const char* fmt, va_list ap );
+ private:
+ int m_addsiz;
};
#endif